首页 > 其他分享 >动态引入图片为什么要是require

动态引入图片为什么要是require

时间:2023-11-17 17:36:06浏览次数:34  
标签:require 编译 webpack 引入 动态 资源 图片

动态的引入图片为什么要使用 require
因为动态添加 src 被当做静态资源处理了,没有进行编译,所以要加上 require

乍一看好像说的很有道理啊,但是仔细一看,这句话说的到底是个啥?针对上面的回答,不禁有如下几个疑问:

什么是静态资源?
为什么动态添加的 src 会被当做的静态的资源?
没有进行编译,是指为是什么没有被编译?
加上 require 为什么能正确的引入资源,是因为加上 require 就能编译了?

1.什么是静态资源

静态资源:一般客户端发送请求到 web 服务器,web 服务器从内存在取到相应的文件,返回给客户端,客户端解析并渲染显示出来。
动态资源:一般客户端请求的动态资源,先将请求交于 web 容器,web 容器连接数据库,数据库处理数据之后,将内容交给 web 服务器,web 服务器返回给客户端解析渲染处理。

静态资源就是直接存放在项目中的资源,这些资源不需要发送专门的请求进行获取。比如 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

可以发现,编译过后的静态地址确实是和 dist 下编译后图片地址是一致的,从而验证想法。
到这里其实就可以解释上面的问题了:动态添加的 src,被编译过后的静态路径为什么无法正确的引入资源?

因为动态的添加的 src 编译过后的地址,与图片资源编译过后的资源地址不一致, 导致无法正确的引入资源
编译过后的 src 地址:../assets/logo.png
编译过后的图片资源地址:/img/logo.6c137b82.png

4. 加上 require 为什么能正确的引入资源,是因为加上 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

上图就是 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,
            },
          },
        },
      ],
    },
  },
};

现在知道 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

因为动态添加的 src,编译过后的文件地址和被编译过后的资源文件地址不一致,从而无法正确引入资源。而使用 require,返回的就是资源文件被编译后的文件地址,从而可以正确的引入资源

6. 问题 3 中,静态的引入一张图片,没有使用 require,为什么返回的依然是编译过后的文件地址?

答:在 webpack 编译的 vue 文件的时候,遇见 src 等属性会默认的使用 require 引入资源路径。引用 vue-cli 官方的一段原话

当在 JavaScript、CSS 或  *.vue  文件中使用相对路径 (必须以  .  开头) 引用一个静态资源时,该资源将会被包含进入 webpack 的依赖图中。在其编译过程中,所有诸如  、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。需要通过绝对路径来引用它们。

9.为什么使用 public 下的资源一定要绝对路径

答:因为虽然 public 文件不会被编译,但是 src 下的文件都会被编译。由于引入的是 public 下的资源,不会走 require,会直接返回代码中的定义的文件地址,该地址无法在编译后的文件目录(dist 目录)下找到对应的文件,会导致引入资源失败。

10.上文件中提到的 webpack,为什么引入资源的时候要有 base64 和打包到 dist 目录下两种的方式,全部打包到的 dist 目录下,他不香吗?

答:为了减少 http 请求。页面中通过路径引入的图片,实际上都会向服务器发送一个请求拿到这张图片。对于资源较小的文件,设置成 base64,既可以减少请求,也不会影响到页面的加载性能。

标签:require,编译,webpack,引入,动态,资源,图片
From: https://www.cnblogs.com/wp-leonard/p/17839299.html

相关文章

  • 动态加载——懒加载实现
    webpack是一个现代JavaScript应用程序的静态模块打包器(modulebundler)。当webpack处理应用程序时,它会递归地构建一个依赖关系图(dependencygraph),其中包含应用程序需要的每个模块,然后将所有这些模块打包成一个或多个bundle。前端工程化演进到今天,webpack做了很大的贡......
  • 纯CSS3实现圆圈动态发光特效动画
    参考文档:https://www.cnblogs.com/cyfeng/p/12625606.htmlhtml文件: <div class="item"></div>css文件:     <!DOCTYPEHTML><html><head><title>纯CSS3实现圆圈动态发光特效动画</title><style>bo......
  • pip生成与安装项目依赖包---提示:No such file or directory: 'requirement.txt'
    错误的原因:安装项目依赖包的文件命令: pipinstall-rrequirement.txt问题:ERROR:Couldnotopenrequirementsfile:[Errno2]Nosuchfileordirectory:'requirement.txt'解决:第一步,检查项目中是否存在文件“requirement.txt”,如果存在,则检查文件名与执行的文件名不一致......
  • Cocos Creator 3.x 如何动态修改3D物体的透明度
    CocosCreator3.x的2DUI有个组件UIOpacity组件可以动态修改UI的透明度,非常方便。很多同学想3D物体上也有一个这样的组件来动态的控制与修改3D物体的透明度。今天基于CocosCreator3.8来实现一个可以动态修改3D物体透明度的组件Opacity3D。 一个3D物体如何才能够半透明显......
  • canvas实现动态替换人物的背景颜色
    起因今天遇见一个特别有意思的小功能。就是更换人物图像的背景颜色。大致操作步骤就是:点击人物-实现背景颜色发生变化将图片绘画到canvas画布上我们需要将图片绘制到canvas画布上。这样做的目的是为了方便我们去操作像素点来更改颜色。首先创建Image的实例。将图片的地......
  • Fastapi框架:引入
    【一】为什么新秀FastAPI火成这样介绍FastAPI有哪些突出特点,浏览官网文档中的Feasures一览【二】FastAPI的突出特点性能优越开发效率提升200%~300%直接减少约40%的人为BUG直观易学易用经简代码/代码重复率低自带API交互文档,开发成果随时交付API开发标准化......
  • @WebServiceClient wsdlLocation 动态给注解内容参数赋值
    动态给注解内容参数赋值@WebServiceClient(name="IXxxService",targetNamespace="http://xxx.xxx.xxx.com",wsdlLocation="${WSDL_URL}")publicclassIXxxServiceextendsService{ //静态变量在静态代码块加载后加载,且注解也在之后加载,完成动态注入修改注解里的参......
  • RequiresPermissions使用方法
    //获得目标方法的签名对象Signaturesignature=point.getSignature();//将目标方法的签名对象转化为MethodSignatureMethodSignaturemethodSignature=(MethodSignature)signature;//获得方法的注解Methodmethod=methodSig......
  • C#调用C++动态库接口函数和回调函数方法 后续
    声明回调委托,C#的委托可以实现C#调用C++的回调,操作函数以后的回调//定义委托,CallingConvention.StdCall可以,CallingConvention.Cdecl不行,参考https://www.it1352.com/1792610.html//[UnmanagedFunctionPointer(CallingConvention.Cdecl)]//不需要要添加该句话,具体参考//htt......
  • iframe本身就不是动态语言,样式和脚本都需要额外导入.iFrame的本质是内联框架的缩写,它
    以下哪个选项的描述是错误的Aiframe是用来在网页中插入第三方页面,早期的页面使用iframe主要是用于导航栏这种很多页面都相同的部分,这样在切换页面的时候避免重复下载Biframe的创建比一般的DOM元素慢了1-2个数量级Ciframe标签会阻塞页面的的加载Diframe本质是动态语言的Inc......