Lassen Sie uns darüber sprechen, warum die dynamische Einführung von Bildern in vue erforderlich ist

52be77932fab1fa96af1655da704e715.jpeg

Ich glaube, dass Freunde, die Vue verwendet haben, vom Interviewer eine solche Frage gestellt bekommen haben: Warum verwenden Sie require, um Bilder dynamisch in Vue zu importieren?

Einige Freunde werden vielleicht verächtlich lächeln: Heh, das ist es, weil die dynamische Hinzufügung von src als statische Ressource behandelt und nicht kompiliert wurde, also muss ich require hinzufügen, und  ich kann es rückwärts aufsagen ...

emmm... Auf den ersten Blick scheint es Sinn zu machen, aber wenn man genau hinschaut, was genau sagt dieser Satz aus? Als Antwort auf die obigen Antworten komme ich nicht umhin, folgende Fragen zu stellen:

  1. Was sind statische Ressourcen?

  2. Warum werden dynamisch hinzugefügte src als statische Ressourcen behandelt?

  3. Nicht kompiliert, bedeutet was nicht kompiliert?

  4. Warum können Ressourcen korrekt mit require importiert werden, obwohl sie mit require kompiliert werden können?

Als ich die letzte Frage hatte, stellte ich fest, dass die obige Antwort etwas zu sagen schien, aber nichts zu sagen schien ... Wenn Sie die gleichen Fragen wie oben haben, lassen Sie es mich Ihnen einzeln sagen

1. Was ist eine statische Ressource?

Es gibt auch eine dynamische Ressource, die der statischen Ressource entspricht.Lassen Sie uns zuerst sehen, wie die großen Jungs im Internet es erklären.

Statische Ressourcen: Im Allgemeinen sendet der Client eine Anfrage an den Webserver, und der Webserver ruft die entsprechende Datei aus dem Speicher ab, sendet sie an den Client zurück, und der Client parst und rendert sie zur Anzeige.

Dynamische Ressourcen: Für dynamische Ressourcen, die vom allgemeinen Client angefordert werden, wird die Anforderung zuerst an den Webcontainer gesendet, und der Webcontainer stellt eine Verbindung zur Datenbank her. Nachdem die Datenbank die Daten verarbeitet hat, wird der Inhalt an den Webserver übergeben und der Der Webserver gibt es zur Analyse und zum Rendern an den Client zurück.

Tatsächlich ist die obige Zusammenfassung bereits sehr klar. Aus Sicht eines Vue-Projekts können wir es einfach so verstehen:

Statische Ressourcen sind Ressourcen, die direkt im Projekt gespeichert sind, und für diese Ressourcen müssen wir keine speziellen Anfragen senden, um sie zu erhalten . Zum Beispiel Bilder, Videos, Audios, Schriftdateien, CSS-Stylesheets usw. im Assets-Verzeichnis.

Dynamische Ressourcen sind Ressourcen, die durch Senden von Anfragen abgerufen werden müssen . Wenn wir beispielsweise Taobao durchsuchen, werden verschiedene Produktinformationen durch das Senden spezieller Anfragen abgerufen, die als dynamische Ressourcen bezeichnet werden können.

2. Warum werden dynamisch hinzugefügte src als statische Ressourcen behandelt?

Bevor wir diese Frage beantworten, müssen wir verstehen, wie ein Browser ein Vue-Projekt ausführen kann.

Wir wissen, dass ein Browser, wenn er eine Webseite öffnet, tatsächlich drei Arten von Dateien ausführt: HTML, CSS und JS. Wenn wir ein Vue-Projekt lokal starten, packen wir das Vue-Projekt tatsächlich zuerst.Der Paketierungsprozess ist der Prozess des Konvertierens und Kompilierens jeder Vue-Datei im Projekt in HTML-, CSS-, JS-Dateien und dann im Browser In Bearbeitung.

Wenn wir require nicht zum Importieren des dynamisch hinzugefügten src verwendet hätten, wie würde es am Ende aussehen?Ich werde Sie zum Experimentieren mitnehmen.

// vue文件中动态引入一张图片
<template>
  <div class="home">
      <!-- 通过v-bind引入资源的方式就称之为动态添加 -->
    <img :src="'../assets/logo.png'" alt="logo">
  </div>
</template>


//最终编译的结果(浏览器上运行的结果)
//这张图片是无法被正确打开的
<img src="../assets/logo.png" alt="logo">

Wir können sehen, dass der dynamisch hinzugefügte src schließlich in eine statische Zeichenfolgenadresse kompiliert wird. Wenn das Programm läuft, importiert es Ressourcen entsprechend dieser Adresse in das Projektverzeichnis. Die Möglichkeit zum Importieren von Ressourcen in das Projektverzeichnis besteht darin, die Ressourcen als statische Ressourcen zu behandeln . Damit ist auch unsere Frage 2 beantwortet.

Einige Freunde könnten verwirrt sein, wenn sie dies sehen. Gibt es ein Problem mit der Adresse, die schließlich zusammengestellt wird? Das Bild in meinem Projekt ist diese Adresse, warum kann es nicht importiert werden? Keine Sorge, schauen wir weiter nach unten.

3. Nicht kompiliert, was bedeutet nicht kompiliert?

Es findet keine Kompilierung statt. Dieser halbe Satz ist sehr verwirrend zu hören. Gemäß Frage 2 wissen wir, dass das dynamisch importierte Bild endgültig kompiliert ist, aber die Bildressource kann nach dem Kompilieren nicht korrekt importiert werden. Dieser Satz ist also falsch. Für unsere Standardantwort schreibe ich sie hier um:

Da das dynamische Hinzufügen von src als statische Ressource behandelt wird und der kompilierte statische Pfad Ressourcen nicht korrekt importieren kann, wird require hinzugefügt

Dann stellt sich hier eine neue Frage: Warum kann der kompilierte statische Pfad Ressourcen nicht korrekt importieren?

Um die Antwort auf diese Frage zu erhalten, müssen wir damit beginnen, ein Bild in der Regel einzuführen. Wenn wir im Projekt ein Bild statisch importieren, kann es auf jeden Fall erfolgreich importiert werden, und die Vue-Datei, in der sich das referenzierte Bild befindet, muss auch kompiliert werden, also in was wird das statisch importierte Bild am Ende kompiliert? Welle:

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


//最终编译的结果
//这张图片是可以被正确打开的
<img src="/img/logo.6c137b82.png" alt="logo">

Gemäß dem obigen Test haben wir festgestellt, dass sich bei Verwendung einer statischen Adresse zum Importieren eines Bildes der Pfad und der Name des Bildes geändert haben und die kompilierte statische Adresse Ressourcen erfolgreich importieren kann. Dies liegt daran, dass standardmäßig alle Dateien im src-Verzeichnis gepackt werden und Bilder unter src ebenfalls in einen neuen Ordner gepackt werden und einen neuen Dateinamen generieren. Die kompilierte statische Adresse leitet die gepackte Bildadresse ein, sodass Ressourcen korrekt referenziert werden können

Ist dies wirklich der Fall? Wir können den Paketierungsbefehl (npm run build) ausführen, um dies zu überprüfen

27a039c22ee3886b6ac022c54dfc96c6.jpeg

Es kann festgestellt werden, dass die kompilierte statische Adresse tatsächlich mit der kompilierten Bildadresse unter dist übereinstimmt, wodurch unsere Idee verifiziert wird.

An dieser Stelle können wir die obige Frage tatsächlich erklären: Warum können der dynamisch hinzugefügte src und der kompilierte statische Pfad Ressourcen nicht korrekt importieren?

Da die kompilierte Adresse der dynamisch hinzugefügten Quelle nicht mit der kompilierten Ressourcenadresse der Bildressource übereinstimmt, kann die Ressource nicht korrekt importiert werden

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

Wie man also die oben genannten Probleme löst, lautet die Antwort: erfordern

4. Warum können Ressourcen korrekt mit require importiert werden, weil sie mit require kompiliert werden können?

Um dieses Problem zu lösen, müssen wir zuerst die zweite Hälfte des Satzes negieren, unabhängig davon, ob require hinzugefügt wird oder nicht, wird ein in die vue-Datei eingeführtes Bild kompiliert.

Dann schauen wir uns require genauer an.

4.1 Was ist erforderlich: Es ist eine Knotenmethode, die verwendet wird, um Module, JSON oder lokale Dateien einzuführen

4.2 Was geschah nach dem Aufruf der require-Methode, um ein Bild einzufügen:

Lassen Sie mich vor der Beantwortung dieser Frage einige inhaltliche Ergänzungen zu Frage 3 vornehmen. Wenn einige Freunde wirklich den Vorgang in Frage 3 befolgen, um dies zu überprüfen, wird geschätzt, dass sie gesprüht werden: Warum unterscheidet sich die endgültige kompilierte Adresse des Bildes, das ich statisch importiert habe, von Ihrer, es ist eine base64, und nach dem Verpacken, es werden unter dist keine neuen Bilder generiert. Wahrscheinlich folgende Situation.

<template>
<div class="home">
    <!-- 直接引入图片静态地址, 不再使用v-bind -->
  <img src="../assets/logo.png" alt="logo">
</div>
</template>


//最终编译的结果
//这张图片是可以被正确打开的
<img src="" alt="logo">

Wir haben oben erwähnt, dass das vue-Projekt schließlich in ein dist-Verzeichnis gepackt wird, was uns also hilft, dieses Paket zu vervollständigen, ja, es ist webpack. Beim Einfügen eines Bildes in das Vue-Projekt werden aufmerksame Schüler feststellen, dass die Adresse des im Browser angezeigten Bildes manchmal eine Base64-Adresse und manchmal eine kompilierte Dateiadresse ist. Das ist der oben beschriebene Unterschied.

Der Grund für diesen Unterschied ist, dass beim Packen von Webpack die Bildressourcen konfiguriert werden. Wir können den folgenden Befehl verwenden, um die Webpack-Konfigurationsdatei im vue-Projekt zur Überprüfung zu generieren:

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

15c5c485a3e31596184285d12c4bd45b.jpeg

Das obige Bild zeigt die standardmäßigen Bildverpackungsregeln von Webpack in Vue. Set type: 'asset', standardmäßig wird für Bilder kleiner als 8k das Bild in base64 konvertiert und direkt in das Bild eingefügt, und es werden keine neuen Bilder im dist-Verzeichnis generiert. Bilder, die größer als 8k sind, werden in das dist-Verzeichnis gepackt und dann wird die Adresse des neuen Bildes an src zurückgegeben.

Das Bild, das ich im obigen Test verwendet habe, ist ein Logobild, das mit vue-cli geliefert wird, und die Größe beträgt 6,69 KB. Gemäß den Standardpaketierungsregeln wird es in base64 konvertiert und in das Bild eingebettet. Der Einfachheit halber habe ich die Standardkonfiguration in vue.config.js geändert. Die Konfiguration lautet wie folgt:

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
            }
          }
        }
      ]
    }
  }
}

So viel wurde oben schon gesagt, was hat das mit verlangen zu tun, natürlich gibt es einen Tropfen.

Wir wissen jetzt, dass vue letztendlich über das Webpack verpackt wird, und eine Reihe von Paketierungsregeln werden in die Webpack-Konfigurationsdatei geschrieben. Die Verpackungsregeln in webpack zielen eigentlich auf einzelne Module ab, mit anderen Worten, webpack verpackt nur Module. Wie behandelt Webpack also das Bild als Modul?Dies erfordert unsere Hauptanforderung.

Wenn wir die require-Methode verwenden, um ein Image einzufügen, behandelt Webpack das Image als Modul und verpackt es gemäß den Regeln in der Konfigurationsdatei. Wir können require als Brücke verwenden, die von der require-Methode eingeführte Ressource verwenden, die Ressource wird als Modul betrachtet und gemäß der Konfigurationsdatei gepackt, und das endgültige Paketierungsergebnis wird zurückgegeben.

Zurück zu Frage 4.2: Was passiert nach dem Aufruf der require-Methode, um ein Bild einzufügen

1. Wenn das Bild kleiner als die im Projekt festgelegte Ressourcengrenzgröße ist, wird die base64 des Bilds zurückgegeben und in den Aufruf der require-Methode eingefügt

2. Wenn das Bild größer als die im Projekt festgelegte Ressourcengrenzgröße ist, wird es in eine neue Bildressource kompiliert. Die require-Methode gibt den neuen Bildressourcenpfad und Dateinamen zurück

Zurück zu Frage 4: Warum das Hinzufügen von require Ressourcen korrekt einführen kann

Da die durch die require-Methode erhaltene Dateiadresse die Dateiadresse ist, nachdem die Ressourcendatei kompiliert wurde (die unter dist generierte Datei oder die base64-Datei), können Sie die entsprechende Datei finden und die Ressource erfolgreich importieren.

Die Antwort ist so einfach, um eine Welle zu verifizieren

// 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">
563fd1899019f1c883b140450402d7a5.jpeg

Gibt es ein Problem, kein Problem. An dieser Stelle können wir auch unsere Standardantwort noch einmal optimieren:

Aufgrund des dynamisch hinzugefügten src stimmt die Adresse der kompilierten Datei nicht mit der Adresse der kompilierten Ressourcendatei überein, sodass Ressourcen nicht korrekt importiert werden können. Und mit require wird die Dateiadresse zurückgegeben, nachdem die Ressourcendatei kompiliert wurde, damit die Ressource korrekt eingeführt werden kann

Angesichts dessen wird geschätzt, dass einige Freunde immer noch Zweifel haben, also werde ich eine weitere Welle erweitern:

5. Warum wird bei Frage 3, statisches Importieren eines Bildes ohne require, immer noch die kompilierte Dateiadresse zurückgegeben?

Antwort: Wenn das Webpack die Vue-Datei kompiliert und auf src und andere Attribute stößt, verwendet es standardmäßig require, um den Ressourcenpfad zu importieren. Zitieren eines Originalsatzes aus dem offiziellen vue-cli

*.vue Wenn Sie mit einem relativen Pfad (muss mit beginnen) in  JavaScript, CSS oder  files auf eine statische Ressource verweisen . , wird die Ressource in das Abhängigkeitsdiagramm des Webpacks aufgenommen. Während der Kompilierung werden alle   Ressourcen-URLs  wie <img src="...">, background: url(...) und CSS  als Modulabhängigkeit aufgelöst .@import

Zum Beispiel url(./image.png) würde übersetzt werden als  require('./image.png'), während:

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

wird kompiliert zu:

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

6. Laut Frage 5 wird beim dynamischen Hinzufügen von src auch require verwendet, um es zu importieren.Warum stimmt die kompilierte Adresse von src nicht mit der Ressourcenadresse von kompilierten Bildressourcen überein?

Antwort: Denn wenn ein Bild dynamisch importiert wird, ist der Attributwert hinter src eigentlich eine Variable. webpack analysiert den Attributwert nach src gemäß der v-bind-Anweisung. Der Ressourcenpfad wird nicht durch reuqire eingeführt. Aus diesem Grund müssen Sie require manuell hinzufügen.

7. Es wird gesagt, dass öffentlich zugängliche Dateien nicht kompiliert werden. Wenn wir also statische Pfade zum Importieren von Ressourcen verwenden, werden wir dann standardmäßig auch require to import verwenden?

Der offizielle Text lautet wie folgt:

Alle  public im Ordner abgelegten statischen Ressourcen werden einfach kopiert, ohne das Webpack zu durchlaufen. Sie müssen mit dem absoluten Pfad darauf verweisen.

Antwort: Nein, die Voraussetzung für die Verwendung von require zum Importieren von Ressourcen ist, dass die Ressource ein Modul ist, das von Webpack analysiert wird, und die Dateien unter public überhaupt nicht kompiliert werden und require nicht verwendet wird.

8. Warum muss der absolute Pfad für Ressourcen unter der Öffentlichkeit verwendet werden?

Antwort: Denn obwohl öffentliche Dateien nicht kompiliert werden, werden Dateien unter src kompiliert. Da es sich bei den importierten Ressourcen um öffentliche Ressourcen handelt, wird require nicht verwendet und die im Code definierte Dateiadresse wird direkt zurückgegeben.Diese Adresse kann die entsprechende Datei im kompilierten Dateiverzeichnis (dist-Verzeichnis) nicht finden, was zu einem Fehler führt Ressourcen importieren.

9. Das im obigen Dokument erwähnte Webpack, warum gibt es zwei Möglichkeiten, Ressourcen zu importieren, base64 und Paketierung in das dist-Verzeichnis, alles in das dist-Verzeichnis gepackt, ist es nicht köstlich?

Antwort: Um HTTP-Anfragen zu reduzieren. Das durch den Pfad in die Seite eingeführte Bild sendet tatsächlich eine Anfrage an den Server, um das Bild zu erhalten. Legen Sie für Dateien mit kleinen Ressourcen base64 fest, wodurch Anforderungen reduziert werden können und die Ladeleistung der Seite nicht beeinträchtigt wird.

  • Referenzlink: cli.vuejs.org/zh/

  • Referenzlink: wjhsh.net/vickylinj-p-15599154.html

Autor: Wie kann man in diesem Alter noch schlafen

https://juejin.cn/post/7159921545144434718

Guess you like

Origin blog.csdn.net/Ed7zgeE9X/article/details/130234049