vue中动态引入图片为什么要是require, 你不知道的那些事
能够发现,编译往后的静态地址确实是和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
上图便是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">
有问题吗,没有问题。到这儿,无妨再对咱们的标准答案进行一次优化:
由于动态增加的src,编译往后的文件地址和被编译往后的资源文件地址不一致,然后无法正确引进资源。而运用require,回来的便是资源文件被编译后的文件地址,然后能够正确的引进资源
看到这,估量还是有一些小伙伴有一些疑问,我再扩展一波:
6. 问题3中,静态的引进一张图片,没有运用require,为什么回来的依然是编译往后的文件地址?
答:在webpack编译的vue文件的时分,遇见src等特点会默许的运用require引进资源途径。引证vue-cli官方的一段原话
当你在 JavaScript、CSS 或
*.vue
文件中运用相对途径 (有必要以.
最初) 引证一个静态资源时,该资源将会被包括进入 webpack 的依靠图中。在其编译过程中,所有诸如<img src="https://juejin.im/post/7159921545144434718/...">
、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,既能够削减恳求,也不会影响到页面的加载功能。
以上便是今日的悉数内容啦,谢谢各位看官老爷的观看。欠好的当地,还请包容。不对的当地,还请纠正。
- 参阅链接:cli.vuejs.org/zh/
- 参阅链接:wjhsh.net/vickylinj-p…
信任用过vue的小伙伴,必定被面试官问过这样一个问题:在vue中动态的引进图片为什么要运用require
有些小伙伴,可能会轻蔑一笑:呵,就这,由于动态增加src被作为静态资源处理了,没有进行编译,所以要加上require, 我倒着都能背出来……
emmm… 乍一看如同说的很有道理啊,可是仔细一看,这句话说的到底是个啥?针对上面的答复,我不由有如下几个疑问:
- 什么是静态资源?
- 为什么动态增加的src会被作为的静态的资源?
- 没有进行编译,是指为是什么没有被编译?
- 加上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)进行验证
能够发现,编译往后的静态地址确实是和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
上图便是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">
有问题吗,没有问题。到这儿,无妨再对咱们的标准答案进行一次优化:
由于动态增加的src,编译往后的文件地址和被编译往后的资源文件地址不一致,然后无法正确引进资源。而运用require,回来的便是资源文件被编译后的文件地址,然后能够正确的引进资源
看到这,估量还是有一些小伙伴有一些疑问,我再扩展一波:
6. 问题3中,静态的引进一张图片,没有运用require,为什么回来的依然是编译往后的文件地址?
答:在webpack编译的vue文件的时分,遇见src等特点会默许的运用require引进资源途径。引证vue-cli官方的一段原话
当你在 JavaScript、CSS 或
*.vue
文件中运用相对途径 (有必要以.
最初) 引证一个静态资源时,该资源将会被包括进入 webpack 的依靠图中。在其编译过程中,所有诸如<img src="https://juejin.im/post/7159921545144434718/...">
、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,既能够削减恳求,也不会影响到页面的加载功能。
以上便是今日的悉数内容啦,谢谢各位看官老爷的观看。欠好的当地,还请包容。不对的当地,还请纠正。
- 参阅链接:cli.vuejs.org/zh/
- 参阅链接:wjhsh.net/vickylinj-p…
能够发现,编译往后的静态地址确实是和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
上图便是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">
有问题吗,没有问题。到这儿,无妨再对咱们的标准答案进行一次优化:
由于动态增加的src,编译往后的文件地址和被编译往后的资源文件地址不一致,然后无法正确引进资源。而运用require,回来的便是资源文件被编译后的文件地址,然后能够正确的引进资源
看到这,估量还是有一些小伙伴有一些疑问,我再扩展一波:
6. 问题3中,静态的引进一张图片,没有运用require,为什么回来的依然是编译往后的文件地址?
答:在webpack编译的vue文件的时分,遇见src等特点会默许的运用require引进资源途径。引证vue-cli官方的一段原话
当你在 JavaScript、CSS 或
*.vue
文件中运用相对途径 (有必要以.
最初) 引证一个静态资源时,该资源将会被包括进入 webpack 的依靠图中。在其编译过程中,所有诸如<img src="https://juejin.im/post/7159921545144434718/...">
、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,既能够削减恳求,也不会影响到页面的加载功能。
以上便是今日的悉数内容啦,谢谢各位看官老爷的观看。欠好的当地,还请包容。不对的当地,还请纠正。
- 参阅链接:cli.vuejs.org/zh/
- 参阅链接:wjhsh.net/vickylinj-p…
信任用过vue的小伙伴,必定被面试官问过这样一个问题:在vue中动态的引进图片为什么要运用require
有些小伙伴,可能会轻蔑一笑:呵,就这,由于动态增加src被作为静态资源处理了,没有进行编译,所以要加上require, 我倒着都能背出来……
emmm… 乍一看如同说的很有道理啊,可是仔细一看,这句话说的到底是个啥?针对上面的答复,我不由有如下几个疑问:
- 什么是静态资源?
- 为什么动态增加的src会被作为的静态的资源?
- 没有进行编译,是指为是什么没有被编译?
- 加上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)进行验证
能够发现,编译往后的静态地址确实是和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
上图便是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">
有问题吗,没有问题。到这儿,无妨再对咱们的标准答案进行一次优化:
由于动态增加的src,编译往后的文件地址和被编译往后的资源文件地址不一致,然后无法正确引进资源。而运用require,回来的便是资源文件被编译后的文件地址,然后能够正确的引进资源
看到这,估量还是有一些小伙伴有一些疑问,我再扩展一波:
6. 问题3中,静态的引进一张图片,没有运用require,为什么回来的依然是编译往后的文件地址?
答:在webpack编译的vue文件的时分,遇见src等特点会默许的运用require引进资源途径。引证vue-cli官方的一段原话
当你在 JavaScript、CSS 或
*.vue
文件中运用相对途径 (有必要以.
最初) 引证一个静态资源时,该资源将会被包括进入 webpack 的依靠图中。在其编译过程中,所有诸如<img src="https://juejin.im/post/7159921545144434718/...">
、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,既能够削减恳求,也不会影响到页面的加载功能。
以上便是今日的悉数内容啦,谢谢各位看官老爷的观看。欠好的当地,还请包容。不对的当地,还请纠正。
- 参阅链接:cli.vuejs.org/zh/
- 参阅链接:wjhsh.net/vickylinj-p…