フロントエンド開発 - Vue が画像を動的にインポートするときに require を使用するのはなぜですか?

作者:この歳でどうやって寝れるの?

https://juejin.cn/post/7159921545144434718

vue を使用したことのある友人は、インタビュアーから次のような質問を受けたに違いないと思います

一部の友人は軽蔑して笑うかもしれません: へー、それだけです。srcの動的追加は静的リソースとして扱われ、コンパイルされていないため、require を追加する必要があり、逆に言えば ...

うーん… 一見、理にかなっているように見えますが、よく見ると、この文は何を言っているのでしょうか。上記の回答に対して、次の質問をせずにはいられません。

  1. 静的リソースとは

  2. 動的に追加された src が静的リソースとして扱われるのはなぜですか?

  3. コンパイルされませんでした。コンパイルされなかったものを意味しますか?

  4. リソースはrequireでコンパイルできるのに、requireで正しくインポートできるのはなぜですか?

最後の質問をしたとき、上記の答えは何かを言っているように見えましたが、何も言っていないように見えました... 上記と同じ質問がある場合は、1つずつ教えてください

1. 静的リソースとは

静的リソースに対応する動的リソースもあります.まず、インターネット上の大物がそれをどのように説明しているかを見てみましょう.

静的リソース: 通常、クライアントは Web サーバーに要求を送信し、Web サーバーはメモリから対応するファイルを取得してクライアントに返し、クライアントはそれを解析して表示用にレンダリングします。

動的リソース: 一般クライアントから要求された動的リソースは、まず Web コンテナーに要求が送信され、Web コンテナーがデータベースに接続し、データベースがデータを処理した後、コンテンツが Web サーバーに渡され、 Web サーバーは、分析とレンダリングのためにそれをクライアントに返します。

実際、上記の要約はすでに非常に明確です。vue プロジェクトの観点からは、単純に次のように理解できます。

静的リソースはプロジェクトに直接保存されるリソースであり、これらのリソースを取得するために特別なリクエストを送信する必要はありませんたとえば、画像、ビデオ、オーディオ、フォント ファイル、css スタイル シートなどは、assets ディレクトリの下にあります。

動的リソースは、リクエストを送信して取得する必要があるリソースですたとえば、タオバオを閲覧すると、動的リソースと呼ばれる特別なリクエストを送信することで、さまざまな製品情報が取得されます。

2. 動的に追加された src が静的リソースとして扱われるのはなぜですか?

この質問に答える前に、ブラウザーが Vue プロジェクトを実行する方法を理解する必要があります。

ブラウザーが Web ページを開くと、実際には html、css、および js の 3 種類のファイルが実行されることがわかっています。ローカルで 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) を実行して確認できます

コンパイルされた静的アドレスは実際に dist の下のコンパイルされたイメージ アドレスと一致していることがわかり、この考えが検証されます。

この時点で、実際に上記の質問を説明できます:動的に追加された src とコンパイルされた静的パスがリソースを正しくインポートできないのはなぜですか?

動的に追加された src のコンパイル済みアドレスがイメージ リソースのコンパイル済みリソース アドレスと一致しないため、リソースを正しくインポートできません

  编译过后的src地址:../assets/logo.png
  编译过后的图片资源地址:/img/logo.6c137b82.png

上記の問題を解決する方法は、次のとおりです。

4. require でリソースを正しくインポートできるのはなぜですか? require でコンパイルできるからですか?

この問題を解決するには、まず文の後半を否定する必要があります.require を追加するかどうかに関係なく、Vue ファイルに導入された画像がコンパイルされます。

次に、require を詳しく見てみましょう。

4.1 何が必要か: モジュール、JSON、またはローカル ファイルを導入するために使用されるノード メソッドです。

4.2 画像を導入するために require メソッドを呼び出した後に何が起こったか:

この質問に答える前に、質問 3 の内容を補足させてください。実際、一部の友人が質問 3 の操作を実際に行って検証した場合、彼らはスプレーされると推定されます。静的にインポートした画像の最終的なコンパイル済みアドレスがあなたのものと異なる理由、それは base64 であり、パッケージング後、 dist new pictures の下には生成されません。おそらく以下の状況です。

// 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 開発 >> webpack.config.development.js

上の図は、vue での webpack のデフォルトのイメージ パッケージ ルールです。セット タイプ: 'asset'。既定では、8k より小さい画像の場合、画像は base64 に変換されて画像に直接挿入され、新しい画像は dist ディレクトリに生成されません。8k を超える画像の場合、それらは dist ディレクトリにパッケージ化され、新しい画像のアドレスが src に返されます。

上記のテストで使用した画像は、vue-cli に付属のロゴ画像で、サイズは 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 のパッケージング ルールは、実際にはモジュール 1 つずつを対象としています。つまり、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">

問題ありますか、問題ありません。この時点で、標準的な回答を再度最適化することもできます。

src が動的に追加されるため、コンパイル済みファイルのアドレスとコンパイル済みリソース ファイルのアドレスが一致しないため、リソースを正しくインポートできません。そして、require を使用すると、リソース ファイルがコンパイルされた後のファイル アドレスが返されるため、リソースを正しく導入できます。

これを見て、一部の友人はまだ疑問を持っていると推定されるため、別の波を展開します。

5. 質問 3 で、require を使用せずに画像を静的にインポートすると、コンパイルされたファイル アドレスがまだ返されるのはなぜですか?

回答: webpack が Vue ファイルをコンパイルするとき、src やその他の属性に遭遇すると、デフォルトで require を使用してリソース パスをインポートします。公式vue-cliの原文引用

JavaScript、CSS、または files で *.vue 相対パス (で始まる必要があります) を使用して 静的リソースを参照すると. 、リソースは webpack の依存関係グラフに含まれます。コンパイル中に、、、   CSS <img src="...">などのすべての リソース URL が モジュールの依存関係として解決されますbackground: url(...)@import

たとえば、次のurl(./image.png) ように翻訳されます require('./image.png')

<img src="./image.png">

次のようにコンパイルされます。

h('img', { attrs: { src: require('./image.png') }})

6. 質問 5 によると、src を動的に追加すると、require も使用してインポートされますが、コンパイルされた src のアドレスと、コンパイルされたイメージ リソースのリソース アドレスが一致しないのはなぜですか?

回答: ピクチャが動的にインポートされる場合、src の背後にある属性値は実際には変数であるためです。webpack は、v-bind 命令に従って、src の後の属性値を解析します。リソース パスは reuqire によって導入されません。これが、require を手動で追加する必要がある理由です。

7. public 配下のファイルはコンパイルされないと言われていますが、静的パスを使用してリソースをインポートする場合、require もデフォルトでインポートするのでしょうか?

公式テキストには次のように書かれています。

public フォルダーに配置された静的リソースは、webpack を介さずに単純にコピーされます。 絶対パスで参照する必要があります。

回答: いいえ、require を使用してリソースをインポートする前提は、リソースが webpack によって解析されるモジュールであり、public の下のファイルはまったくコンパイルされず、require が使用されないことです。

8. public の下のリソースに絶対パスを使用しなければならない理由

回答: public ファイルはコンパイルされませんが、src の下のファイルはコンパイルされるためです。インポートされたリソースはパブリック リソースであるため、require は使用されず、コードで定義されたファイル アドレスが直接返されます.このアドレスは、コンパイルされたファイル ディレクトリ (dist ディレクトリ) で対応するファイルを見つけることができず、失敗の原因となります.リソースをインポートします。

9. 上記のドキュメントに記載されている webpack では、リソースをインポートする方法が base64 と dist ディレクトリへのパッケージ化の 2 つあるのに、すべて dist ディレクトリにパッケージ化されているのはなぜですか?

回答: http リクエストを減らすためです。ページ内のパスによって導入された画像は、実際に画像を取得するためにサーバーに要求を送信します。リソースが小さいファイルの場合は、base64 に設定します。これにより、リクエストを減らすことができ、ページの読み込みパフォーマンスには影響しません。

  • 参考リンク:cli.vuejs.org/zh/

  • 参考リンク:wjhsh.net/vickylinj-p-15599154.html

おすすめ

転載: blog.csdn.net/helloyangkl/article/details/129085695