相信用過vue的小伙伴算墨,肯定被面試官問過這樣一個問題:在vue中動態(tài)的引入圖片為什么要使用require蹲嚣?
有些小伙伴赚瘦,可能會輕蔑一笑:呵车摄,就這寺谤,因為動態(tài)添加src被當(dāng)做靜態(tài)資源處理了,沒有進(jìn)行編譯吮播,所以要加上require变屁, 我倒著都能背出來......
emmm... 乍一看好像說的很有道理啊,但是仔細(xì)一看意狠,這句話說的到底是個啥粟关?針對上面的回答羞海,我不禁有如下幾個疑問:
1伐蒋、什么是靜態(tài)資源?
2蛹头、為什么動態(tài)添加的src會被當(dāng)做的靜態(tài)的資源 院塞?
3蛔垢、沒有進(jìn)行編譯,是指為什么沒有被編譯迫悠?
4鹏漆、加上require為什么能正確的引入資源,是因為加上require就能編譯了创泄?
當(dāng)我產(chǎn)生最后一個疑問的時候艺玲,發(fā)現(xiàn)上面的答案看似說了些啥,但好像又什么都沒說...... 如果各位看官老爺也有如上幾個疑問鞠抑,那就讓我給大家一一解惑
1.什么是靜態(tài)資源
與靜態(tài)資源相對應(yīng)的還有一個動態(tài)資源饭聚,先讓我們看看網(wǎng)上的各位大佬們怎么解釋的。
靜態(tài)資源:一般客戶端發(fā)送請求到web服務(wù)器搁拙,web服務(wù)器從內(nèi)存取到相應(yīng)的文件秒梳,返回給客戶端法绵,客戶端解析并渲染顯示出來。
動態(tài)資源:一般客戶端請求的動態(tài)資源酪碘,先將請求交于web服務(wù)器朋譬,web服務(wù)器連接數(shù)據(jù)庫,數(shù)據(jù)庫處理數(shù)據(jù)之后兴垦,將內(nèi)容交給web服務(wù)器徙赢,web服務(wù)器返回給客戶端解析渲染處理。
其實(shí)上面的總結(jié)已經(jīng)很清晰了探越。站在一個vue項目的角度狡赐,我們可以簡單的理解為:
靜態(tài)資源就是直接存放在項目中的資源,這些資源不需要我們發(fā)送專門的請求進(jìn)行獲取钦幔。比如assets目錄下面的圖片枕屉,視頻,音頻鲤氢,字體文件搀擂,css樣式表等。
動態(tài)資源就是需要發(fā)送請求獲取到的資源铜异。比如我們刷淘寶的時候哥倔,不同的商品信息是發(fā)送的專門的請求獲取到的秸架,就可以稱之為動態(tài)資源揍庄。
2. 為什么動態(tài)添加的src會被當(dāng)做的靜態(tài)的資源?
回答這個問題之前东抹,我們需要了解一下蚂子,瀏覽器是怎么能運(yùn)行一個vue項目的。
我們知道瀏覽器打開一個網(wǎng)頁缭黔,實(shí)際上運(yùn)行的是html食茎,css,js三種類型的文件馏谨。當(dāng)我們本地啟動一個vue項目的時候别渔,實(shí)際上是先將vue項目進(jìn)行打包,打包的過程就是將項目中的一個個vue文件轉(zhuǎn)編譯成html惧互,css哎媚,js文件的過程,而后再在瀏覽器上運(yùn)行的喊儡。
那動態(tài)添加的src如果我們沒有使用require引入拨与,最終會打包成什么樣子呢,我?guī)Т蠹覍?shí)驗一波艾猜。
// vue文件中動態(tài)引入一張圖片
<template>
<div class="home">
<!-- 通過v-bind引入資源的方式就稱之為動態(tài)添加 -->
<img :src="'../assets/logo.png'" alt="logo">
</div>
</template>
//最終編譯的結(jié)果(瀏覽器上運(yùn)行的結(jié)果)
//這張圖片是無法被正確打開的
<img src="../assets/logo.png" alt="logo">
我們可以看出买喧,動態(tài)添加的src最終會編譯成一個靜態(tài)的字符串地址捻悯。程序運(yùn)行的時候,會按照這個地址去項目目錄中引入資源淤毛。而 去項目目錄中引入資源的這種方式今缚,就是將該資源當(dāng)成了靜態(tài)資源。所以這也就回答了我們的問題2钱床。
看到這里估計就有小伙伴疑惑了荚斯,這個最終被編譯的地址有什么問題嗎?我項目中的圖片就是這個地址查牌,為什么無法引入事期?別急,我們繼續(xù)往下看纸颜。
3. 沒有進(jìn)行編譯兽泣,是指的是什么沒有被編譯?
沒有進(jìn)行編譯胁孙。這半句話唠倦,就聽得很讓人懵逼了。按照問題2我們知道這個動態(tài)引入的圖片最終是被編譯了涮较,只是被編譯之后無法正確的引入圖片資源而已稠鼻。所以這句話本來就是錯的。針對于我們的標(biāo)準(zhǔn)答案狂票,我在這里進(jìn)行改寫:
因為動態(tài)添加src被當(dāng)做靜態(tài)資源處理了候齿,而被編譯過后的靜態(tài)路徑無法正確的引入資源,所以要加上require
那這里就誕生了一個新的疑問:被編譯過后的靜態(tài)路徑為什么無法正確的引入資源闺属?
想得到這個問題的答案慌盯,我們得先從正常的引入一張圖片開始。在項目中我們靜態(tài)的引入一張圖片肯定是可以引入成功的掂器,而引用圖片所在的vue文件肯定也是被編譯的亚皂,那靜態(tài)引入圖片最終會被編譯成什么樣呢,模擬一波:
// vue文件中靜態(tài)的引入一張圖片
<template>
<div class="home">
<!-- 直接引入圖片靜態(tài)地址国瓮, 不再使用v-bind -->
<img src="../assets/logo.png" alt="logo">
</div>
</template>
//最終編譯的結(jié)果
//這張圖片是可以被正確打開的
<img src="/img/logo.6c137b82.png" alt="logo">
根據(jù)上面的測試灭必,我們發(fā)現(xiàn),使用靜態(tài)的地址去引入一張圖片乃摹,圖片的路徑和圖片的名稱已經(jīng)發(fā)生了改變禁漓,并且編譯后過后的靜態(tài)地址是可以成功的引入資源的。這是因為峡懈,在默認(rèn)情況下璃饱,src目錄下面的所有文件都會被打包,src下面的圖片也會被打包在新的文件夾下并生成新的文件名肪康。編譯過后的靜態(tài)地址引入的是打包過后的圖片地址荚恶,從而可以正確的引用資源
事實(shí)確實(shí)是這樣嗎撩穿?我們可以執(zhí)行打包命令(npm run build)進(jìn)行驗證
可以發(fā)現(xiàn),編譯過后的靜態(tài)地址確實(shí)是和dist下編譯后圖片地址是一致的谒撼,從而驗證我們的想法食寡。
到這里我們其實(shí)就可以解釋上面的問題了:動態(tài)添加的src,被編譯過后的靜態(tài)路徑為什么無法正確的引入資源廓潜?
因為動態(tài)的添加的src編譯過后的地址抵皱,與圖片資源編譯過后的資源地址不一致, 導(dǎo)致無法正確的引入資源
編譯前的src地址:../assets/logo.png
編譯過后的圖片資源地址:/img/logo.6c137b82.png
那要怎么解決上述的問題呢辩蛋,答案就是:require
4. 加上require為什么能正確的引入資源呻畸,是因為加上require就能編譯了?
針對這個問題悼院,首先就要否定后半句伤为,無論加不加require,vue文件中引入一張圖片都會被編譯据途。
接著我們再來好好了解一下绞愚,require。
require是什么: 是一個node方法颖医,用于引入模塊位衩,JSON或本地文件
調(diào)用require方法引入一張圖片之后發(fā)生了什么:
在回答這個問題之前,容我先對問題3中的內(nèi)容進(jìn)行一定的補(bǔ)充熔萧。其實(shí)如果真的有小伙伴跟著問題三中的操作進(jìn)行驗證糖驴,估計就要開噴了:為什么我靜態(tài)引入的圖片最終編譯的地址和你的不一樣,是個base64哪痰,而且打包之后dist下面也沒有生成新的圖片遂赠。大概就是下面這樣的情況久妆。
// vue文件中靜態(tài)的引入一張圖片
<template>
<div class="home">
<!-- 直接引入圖片靜態(tài)地址晌杰, 不再使用v-bind -->
<img src="../assets/logo.png" alt="logo">
</div>
</template>
//最終編譯的結(jié)果
//這張圖片是可以被正確打開的
<img src="" alt="logo">
先別急著噴,實(shí)際上造成這種差異的原因筷弦,是因為我改了一下webpack中的配置肋演。接下來涉及少量webpack代碼,不了解webpack的小伙伴也沒關(guān)系烂琴,了解原理即可爹殊。
在上文中的我們提到,vue項目最終會被打包成一個dist目錄奸绷,那么是什么幫我們完成這個打包的呢梗夸,沒錯,就是webpack号醉。在vue項目中的引入一張圖片的時候反症,細(xì)心的同學(xué)會發(fā)現(xiàn)辛块,有的時候,瀏覽器上顯示圖片地址是一個base64铅碍,有的時候润绵,是一個被編譯過后的文件地址。也就是上述描述的差異胞谈。
之所以會造成這種差異尘盼,是webpack打包的時候,對圖片資源進(jìn)行了相關(guān)的配置烦绳。我們可以通過如下命令生成vue項目中的webpack配置文件卿捎,進(jìn)行驗證:
npx vue-cli-service inspect --mode development >> webpack.config.development.js
上圖就是vue中webpack默認(rèn)的圖片打包規(guī)則。設(shè)置 type: 'asset'径密,默認(rèn)的娇澎,對于小于8k的圖片,會將圖片轉(zhuǎn)成base64 直接插入圖片睹晒,不會再在dist目錄生成新圖片趟庄。對于大于8k的圖片,會打包進(jìn)dist目錄伪很,之后將新圖片地址返回給src戚啥。
而我在上述測試中使用的圖片,是vue-cli自帶的一張logo圖片锉试,大小是6.69k猫十。按照默認(rèn)的打包規(guī)則,是會轉(zhuǎn)成base64呆盖,嵌入圖片中的拖云。所以為了講述方便,我在vue.config.js中修改了其默認(rèn)的配置应又,配置如下:
module.exports = {
// 使用configureWebpack對象宙项,下面可以直接按照webpack中的寫法進(jìn)行編寫
// 編寫的內(nèi)容,最終會被webpack-merge插件合并到webpack.config.js主配置文件中
configureWebpack: {
module: {
rules: [
{
test: /\.(png|jpe?g|gif|webp|avif)(\?.*)?$/,
type: 'asset',
parser: {
dataUrlCondition: {
// 這里我將默認(rèn)的大小限制改成6k株扛。
// 當(dāng)圖片小于6k時候尤筐,使用base64引入圖片;大于6k時洞就,打包到dist目錄下再進(jìn)行引入
maxSize: 1024 * 6
}
}
}
]
}
}
}
那上面說了這么多盆繁,和require有啥關(guān)系,自然是有滴旬蟋。
我們現(xiàn)在知道vue最終是通過webpack打包油昂,并且會在webpack配置文件中編寫一系列打包規(guī)則。而webpack中的打包規(guī)則,針對的其實(shí)是一個一個模塊冕碟,換而言之webpack只會對模塊進(jìn)行打包稠腊。那webpack怎么將圖片當(dāng)成一個模塊呢,這就要用到我們的正主require鸣哀。
當(dāng)我們使用require方法引入一張圖片的時候架忌,webpack會將這張圖片當(dāng)成一個模塊,并根據(jù)配置文件中的規(guī)則進(jìn)行打包我衬。我們可以將require當(dāng)成一個橋梁叹放,使用了require方法引入的資源,該資源就會當(dāng)成模塊并根據(jù)配置文件進(jìn)行打包挠羔,并返回最終的打包結(jié)果井仰。
回到問題4.2:調(diào)用require方法引入一張圖片之后發(fā)生了什么
1.如果這張圖片小于項目中設(shè)置的資源限制大小,則會返回圖片的base64插入到require方法的調(diào)用處
2.如果這張圖片大于項目中設(shè)置的資源限制大小破加,則會將這個圖片編譯成一個新的圖片資源俱恶。require方法返回新的圖片資源路徑及文件名
回到問題4:為什么加上require能正確的引入資源
因為通過require方法拿到的文件地址,是資源文件編譯過后的文件地址(dist下生成的文件或base64文件)范舀,因此可以找對應(yīng)的文件合是,從而成功引入資源。
答案就是這么簡單锭环,來驗證一波
// vue文件中使用require動態(tài)的引入一張圖片
<template>
<div class="home">
<!-- 使用require動態(tài)引入圖片 -->
<img :src="require('../assets/logo.png')" alt="logo">
</div>
</template>
//最終編譯的結(jié)果
//這張圖片是可以被正確打開的
<img src="/img/logo.6c137b82.png" alt="logo">
有問題嗎聪全,沒有問題。到這里辅辩,不妨再對我們的標(biāo)準(zhǔn)答案進(jìn)行一次優(yōu)化:
因為動態(tài)添加的src难礼,編譯過后的文件地址和被編譯過后的資源文件地址不一致,從而無法正確引入資源玫锋。而使用require蛾茉,返回的就是資源文件被編譯后的文件地址,從而可以正確的引入資源
看到這撩鹿,估計還是有一些小伙伴有一些疑問谦炬,我再擴(kuò)展一波:
5. 問題3中,靜態(tài)的引入一張圖片三痰,沒有使用require吧寺,為什么返回的依然是編譯過后的文件地址窜管?
答:在webpack編譯的vue文件的時候散劫,遇見src等屬性會默認(rèn)的使用require引入資源路徑。引用vue-cli官方的一段原話
當(dāng)你在 JavaScript幕帆、CSS 或 *.vue 文件中使用相對路徑 (必須以 . 開頭) 引用一個靜態(tài)資源時获搏,該資源將會被包含進(jìn)入 webpack 的依賴圖中。在其編譯過程中,所有諸如 <img src="...">常熙、background: url(...) 和 CSS @import 的資源 URL 都會被解析為一個模塊依賴纬乍。
例如,url(./image.png) 會被翻譯為 require('./image.png')裸卫,而:
<img src="./image.png">
將會被編譯到:
h('img', { attrs: { src: require('./image.png') }})
6. 按照問題5中所說仿贬,那么動態(tài)添加src的時候也會使用require引入,為什么src編譯過后的地址墓贿,與圖片資源編譯過后的資源地址不一致
答:因為動態(tài)引入一張圖片的時候茧泪,src后面的屬性值,實(shí)際上是一個變量聋袋。webpack會根據(jù)v-bind指令去解析src后面的屬性值队伟。并不會通過reuqire引入資源路徑。這也是為什么需要手動的添加require幽勒。
7.據(jù)說public下面的文件不會被編譯嗜侮,那我們使用靜態(tài)路徑去引入資源的時候,也會默認(rèn)的使用require引入嗎啥容?
官方的原文是這樣子的:
任何放置在 public 文件夾的靜態(tài)資源都會被簡單的復(fù)制锈颗,而不經(jīng)過 webpack。你需要通過絕對路徑來引用它們咪惠。
答:不會宜猜,使用require引入資源的前提的該資源是webpack解析的模塊,而public下的文件壓根就不會走編譯硝逢,也就不會使用到require姨拥。
8.為什么使用public下的資源一定要絕對路徑
答:因為雖然public文件不會被編譯,但是src下的文件都會被編譯渠鸽。由于引入的是public下的資源叫乌,不會走require,會直接返回代碼中的定義的文件地址徽缚,該地址無法在編譯后的文件目錄(dist目錄)下找到對應(yīng)的文件憨奸,會導(dǎo)致引入資源失敗。
9.上文件中提到的webpack凿试,為什么引入資源的時候要有base64和打包到dist目錄下兩種的方式排宰,全部打包到的dist目錄下,他不香嗎那婉?
答:為了減少http請求板甘。頁面中通過路徑引入的圖片,實(shí)際上都會向服務(wù)器發(fā)送一個請求拿到這張圖片详炬。對于資源較小的文件盐类,設(shè)置成base64,既可以減少請求,也不會影響到頁面的加載性能在跳。