vue中動態引入圖片為什麼要是require, 你不知道的那些事

来源:https://www.cnblogs.com/lisongming/archive/2022/10/29/16839892.html
-Advertisement-
Play Games

相信用過vue的小伙伴,肯定被面試官問過這樣一個問題:在vue中動態的引入圖片為什麼要使用require 有些小伙伴,可能會輕蔑一笑:呵,就這,因為動態添加src被當做靜態資源處理了,沒有進行編譯,所以要加上require, 我倒著都能背出來...... emmm... 乍一看好像說的很有道理啊... ...


相信用過vue的小伙伴,肯定被面試官問過這樣一個問題:在vue中動態的引入圖片為什麼要使用require

有些小伙伴,可能會輕蔑一笑:呵,就這,因為動態添加src被當做靜態資源處理了,沒有進行編譯,所以要加上require, 我倒著都能背出來......

emmm... 乍一看好像說的很有道理啊,但是仔細一看,這句話說的到底是個啥?針對上面的回答,我不禁有如下幾個疑問:

  1. 什麼是靜態資源?
  2. 為什麼動態添加的src會被當做的靜態的資源?
  3. 沒有進行編譯,是指為是什麼沒有被編譯?
  4. 加上require為什麼能正確的引入資源,是因為加上require就能編譯了?

當我產生最後一個疑問的時候,發現上面的答案看似說了些啥,但好像又什麼都沒說...... 如果各位看官老爺也有如上幾個疑問,那就讓我給大家一一解惑

1.什麼是靜態資源

與靜態資源相對應的還有一個動態資源,先讓我們看看網上的各位大佬們怎麼解釋的。

靜態資源:一般客戶端發送請求到web伺服器,web伺服器從記憶體在取到相應的文件,返回給客戶端,客戶端解析並渲染顯示出來。

動態資源:一般客戶端請求的動態資源,先將請求交於web容器,web容器連接資料庫,資料庫處理數據之後,將內容交給web伺服器,web伺服器返回給客戶端解析渲染處理。

其實上面的總結已經很清晰了。站在一個vue項目的角度,我們可以簡單的理解為:

靜態資源就是直接存放在項目中的資源,這些資源不需要我們發送專門的請求進行獲取。比如assets目錄下麵的圖片,視頻,音頻,字體文件,css樣式表等。

動態資源就是需要發送請求獲取到的資源。比如我們刷淘寶的時候,不同的商品信息是發送的專門的請求獲取到的,就可以稱之為動態資源。

2. 為什麼動態添加的src會被當做的靜態的資源?

回答這個問題之前,我們需要瞭解一下,瀏覽器是怎麼能運行一個vue項目的。

我們知道瀏覽器打開一個網頁,實際上運行的是html,css,js三種類型的文件。當我們本地啟動一個vue項目的時候,實際上是先將vue項目進行打包,打包的過程就是將項目中的一個個vue文件轉編譯成html,css,js文件的過程,而後再在瀏覽器上運行的。

那動態添加的src如果我們沒有使用require引入,最終會打包成什麼樣子呢,我帶大家實驗一波。

// vue文件中動態引入一張圖片
<template>
  <div class="home">
      <!-- 通過v-bind引入資源的方式就稱之為動態添加 -->
    <img :src="'../assets/logo.png'" alt="logo">
  </div>
</template>

//最終編譯的結果(瀏覽器上運行的結果)
//這張圖片是無法被正確打開的
<img src="../assets/logo.png" alt="logo">  

我們可以看出,動態添加的src最終會編譯成一個靜態的字元串地址。程式運行的時候,會按照這個地址去項目目錄中引入資源。而 去項目目錄中引入資源的這種方式,就是將該資源當成了靜態資源。所以這也就回答了我們的問題2。

看到這裡估計就有小伙伴疑惑了,這個最終被編譯的地址有什麼問題嗎?我項目中的圖片就是這個地址,為什麼無法引入?別急,我們繼續往下看。

3. 沒有進行編譯,是指的是什麼沒有被編譯?

沒有進行編譯。這半句話,就聽得很讓人懵逼了。按照問題2我們知道這個動態引入的圖片最終是被編譯了,只是被編譯之後無法正確的引入圖片資源而已。所以這句話本來就是錯的。針對於我們的標準答案,我在這裡進行改寫:

因為動態添加src被當做靜態資源處理了,而被編譯過後的靜態路徑無法正確的引入資源,所以要加上require

那這裡就誕生了一個新的疑問:被編譯過後的靜態路徑為什麼無法正確的引入資源?

想得到這個問題的答案,我們得先從正常的引入一張圖片開始。在項目中我們靜態的引入一張圖片肯定是可以引入成功的,而引用圖片所在的vue文件肯定也是被編譯的,那靜態引入圖片最終會被編譯成什麼樣呢,模擬一波:

// vue文件中靜態的引入一張圖片
<template>
  <div class="home">
      <!-- 直接引入圖片靜態地址, 不再使用v-bind -->
    <img src="../assets/logo.png" alt="logo">
  </div>
</template>

//最終編譯的結果
//這張圖片是可以被正確打開的
<img src="/img/logo.6c137b82.png" alt="logo">

根據上面的測試,我們發現,使用靜態的地址去引入一張圖片,圖片的路徑和圖片的名稱已經發生了改變,並且編譯後過後的靜態地址是可以成功的引入資源的。這是因為,在預設情況下,src目錄下麵的所有文件都會被打包,src下麵的圖片也會被打包在新的文件夾下並生成新的文件名。編譯過後的靜態地址引入的是打包過後的圖片地址,從而可以正確的引用資源

事實確實是這樣嗎?我們可以執行打包命令(npm run build)進行驗證

image.png

可以發現,編譯過後的靜態地址確實是和dist下編譯後圖片地址是一致的,從而驗證我們的想法。

到這裡我們其實就可以解釋上面的問題了:動態添加的src,被編譯過後的靜態路徑為什麼無法正確的引入資源?

因為動態的添加的src編譯過後的地址,與圖片資源編譯過後的資源地址不一致, 導致無法正確的引入資源

  編譯過後的src地址:../assets/logo.png
  編譯過後的圖片資源地址:/img/logo.6c137b82.png

那要怎麼解決上述的問題呢,答案就是:require

4. 加上require為什麼能正確的引入資源,是因為加上require就能編譯了?

針對這個問題,首先就要否定後半句,無論加不加require,vue文件中引入一張圖片都會被編譯。

接著我們再來好好瞭解一下,require。

4.1 require是什麼: 是一個node方法,用於引入模塊,JSON或本地文件

4.2 調用require方法引入一張圖片之後發生了什麼:

在回答這個問題之前,容我先對問題3中的內容進行一定的補充。其實如果真的有小伙伴跟著問題三中的操作進行驗證,估計就要開噴了:為什麼我靜態引入的圖片最終編譯的地址和你的不一樣,是個base64,而且打包之後dist下麵也沒有生成新的圖片。大概就是下麵這樣的情況。

// vue文件中靜態的引入一張圖片
<template>
<div class="home">
    <!-- 直接引入圖片靜態地址, 不再使用v-bind -->
  <img src="../assets/logo.png" alt="logo">
</div>
</template>

//最終編譯的結果
//這張圖片是可以被正確打開的
<img src="" alt="logo">

先別急著噴,實際上造成這種差異的原因,是因為我改了一下webpack中的配置。接下來涉及少量webpack代碼,不瞭解webpack的小伙伴也沒關係,瞭解原理即可。

在上文中的我們提到,vue項目最終會被打包成一個dist目錄,那麼是什麼幫我們完成這個打包的呢,沒錯,就是webpack。在vue項目中的引入一張圖片的時候,細心的同學會發現,有的時候,瀏覽器上顯示圖片地址是一個base64,有的時候,是一個被編譯過後的文件地址。也就是上述描述的差異。

之所以會造成這種差異,是webpack打包的時候,對圖片資源進行了相關的配置。我們可以通過如下命令生成vue項目中的webpack配置文件,進行驗證:

npx vue-cli-service inspect --mode development >> webpack.config.development.js

image.png

上圖就是vue中webpack預設的圖片打包規則。設置 type: 'asset',預設的,對於小於8k的圖片,會將圖片轉成base64 直接插入圖片,不會再在dist目錄生成新圖片。對於大於8k的圖片,會打包進dist目錄,之後將新圖片地址返回給src。

而我在上述測試中使用的圖片,是vue-cli自帶的一張logo圖片,大小是6.69k。按照預設的打包規則,是會轉成base64,嵌入圖片中的。所以為了講述方便,我在vue.config.js中修改了其預設的配置,配置如下:

module.exports = {
    // 使用configureWebpack對象,下麵可以直接按照webpack中的寫法進行編寫
    // 編寫的內容,最終會被webpack-merge插件合併到webpack.config.js主配置文件中
  configureWebpack: { 
    module: {
      rules: [
        {
          test: /\.(png|jpe?g|gif|webp|avif)(\?.*)?$/,
          type: 'asset',
          parser: {
            dataUrlCondition: {
             // 這裡我將預設的大小限制改成6k。
              // 當圖片小於6k時候,使用base64引入圖片;大於6k時,打包到dist目錄下再進行引入
              maxSize: 1024 * 6
            }
          }
        }
      ]
    }
  }
}

那上面說了這麼多,和require有啥關係,自然是有滴。

我們現在知道vue最終是通過webpack打包,並且會在webpack配置文件中編寫一系列打包規則。而webpack中的打包規則,針對的其實是一個一個模塊,換而言之webpack只會對模塊進行打包。那webpack怎麼將圖片當成一個模塊呢,這就要用到我們的正主require。

當我們使用require方法引入一張圖片的時候,webpack會將這張圖片當成一個模塊,並根據配置文件中的規則進行打包。我們可以將require當成一個橋梁,使用了require方法引入的資源,該資源就會當成模塊並根據配置文件進行打包,並返回最終的打包結果。

回到問題4.2:調用require方法引入一張圖片之後發生了什麼

1.如果這張圖片小於項目中設置的資源限制大小,則會返回圖片的base64插入到require方法的調用處

2.如果這張圖片大於項目中設置的資源限制大小,則會將這個圖片編譯成一個新的圖片資源。require方法返回新的圖片資源路徑及文件名

回到問題4:為什麼加上require能正確的引入資源

因為通過require方法拿到的文件地址,是資源文件編譯過後的文件地址(dist下生成的文件或base64文件),因此可以找對應的文件,從而成功引入資源。

答案就是這麼簡單,來驗證一波

// vue文件中使用require動態的引入一張圖片
<template>
  <div class="home">
      <!-- 使用require動態引入圖片 -->
      <img :src="require('../assets/logo.png')" alt="logo">
  </div>
</template>

//最終編譯的結果
//這張圖片是可以被正確打開的
<img src="/img/logo.6c137b82.png" alt="logo">

image.png

有問題嗎,沒有問題。到這裡,不妨再對我們的標準答案進行一次優化:

因為動態添加的src,編譯過後的文件地址和被編譯過後的資源文件地址不一致,從而無法正確引入資源。而使用require,返回的就是資源文件被編譯後的文件地址,從而可以正確的引入資源

看到這,估計還是有一些小伙伴有一些疑問,我再擴展一波:

6. 問題3中,靜態的引入一張圖片,沒有使用require,為什麼返回的依然是編譯過後的文件地址?

答:在webpack編譯的vue文件的時候,遇見src等屬性會預設的使用require引入資源路徑。引用vue-cli官方的一段原話

當你在 JavaScript、CSS 或 *.vue 文件中使用相對路徑 (必須以 . 開頭) 引用一個靜態資源時,該資源將會被包含進入 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') }})

7. 按照問題6中所說,那麼動態添加src的時候也會使用require引入,為什麼src編譯過後的地址,與圖片資源編譯過後的資源地址不一致

答:因為動態引入一張圖片的時候,src後面的屬性值,實際上是一個變數。webpack會根據v-bind指令去解析src後面的屬性值。並不會通過reuqire引入資源路徑。這也是為什麼需要手動的添加require。

8.據說public下麵的文件不會被編譯,那我們使用靜態路徑去引入資源的時候,也會預設的使用require引入嗎?

官方的原文是這樣子的:

任何放置在 public 文件夾的靜態資源都會被簡單的複製,而不經過 webpack。你需要通過絕對路徑來引用它們。

答:不會,使用require引入資源的前提的該資源是webpack解析的模塊,而public下的文件壓根就不會走編譯,也就不會使用到require。

9.為什麼使用public下的資源一定要絕對路徑

答:因為雖然public文件不會被編譯,但是src下的文件都會被編譯。由於引入的是public下的資源,不會走require,會直接返回代碼中的定義的文件地址,該地址無法在編譯後的文件目錄(dist目錄)下找到對應的文件,會導致引入資源失敗。

10.上文件中提到的webpack,為什麼引入資源的時候要有base64和打包到dist目錄下兩種的方式,全部打包到的dist目錄下,他不香嗎?

答:為了減少http請求。頁面中通過路徑引入的圖片,實際上都會向伺服器發送一個請求拿到這張圖片。對於資源較小的文件,設置成base64,既可以減少請求,也不會影響到頁面的載入性能。

以上就是今天的全部內容啦,謝謝各位看官老爺的觀看。不好的地方,還請包涵。不對的地方,還請指正。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 本篇文章主要介紹的調用微信公眾號和小程式之後的開發流程,主要以實現步驟為主。 前提準備 已經申請了微信公眾號和小程式(由於官方文檔寫的很詳細,這裡就不在進行說明瞭); 微信公眾號和小程式的進行關聯了(主要是為了消息推送); 小程式的開發流程 提前註冊微信小程式,註冊地址:https://mp. ...
  • 因為工作需要,需要將Unity項目生成Android工程導入到Android Studio中,其中主要遇到的問題: 1.缺乏依賴包 2.依賴包重覆 有關問題1:詳細問題太多,這裡不贅述,可能涉及庫版本相容等問題,可自行百度解決。 有關問題2:這個問題也是本隨筆的主要問題,我遇到的主要是對於Volle ...
  • 1.1 初始化腳手架 1.1.1 說明 Vue 腳手架是 Vue 官方提供的標準化開發工具(開發平臺) 最新的版本是 4.x 文檔 1.1.2 具體步驟 第一步(僅第一次執行):全局安裝 @vue/cli npm install -g @vue/cli 第二步: 切換到要創建項目的目錄 ,然後使用命 ...
  • Day2 和前一天的頁面一樣, 但是改了一些規範, 並且加上了彈出框(簡陋版). 主要是改了首尾的引航欄, 將原來的直接使用a標簽變成使用ul>(li>a)標簽 成果 修改了首尾的引航欄, 將原來的直接使用a標簽變成使用ul>(li>a)標簽: 將ul>li標簽通過css的display: inli ...
  • 使用 Notification 可以在系統級別發送頁面外部顯示的桌面通知,即使瀏覽器在後臺運行也可以向用戶發出消息 檢查許可權 發送通知需要用戶授權,通過只讀屬性 Notification.permission 獲取是否有許可權,其屬性值有 granted:用戶已明確授權顯示通知的許可權 denied:用 ...
  • 一、瀏覽器的渲染過程 註意:這個過程是逐步完成的,為了更好的用戶體驗,渲染引擎將會儘可能早的將內容呈現到屏幕上,並不會等到所有的html 都解析完成之後再去構建和佈局 render 樹。它是解析完一部分內容就顯示一部分內容,同時,可能還在通過網路下載其餘內容。 渲染過程解讀如下: 首先解析收到的文檔 ...
  • 一、函數 數的定義有兩種:一為函數的聲明、二為函數表達式-匿名函數 函數聲明 1 function fn(){//需要函數名 2 console.log("函數聲明"); 3 } 4 fn() 函數表達式 1 const fs=function(){//需要定義變數接收 2 console.log( ...
  • Day1 根據自己的垃圾html+css水平寫出了下麵這個頁面: 首先, 首部的超鏈接都能夠鏈接上, 但是不會[更多]的彈出視窗, 右面的天氣/設置/個人也沒有做, 不會怎麼去調用 中間的圖片和搜索欄能夠正常使用, 搜索欄能夠正常搜索. 但是無法水平垂直居中, 而且, 兩個input無法靠近在一起. ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...