vue에 이미지를 동적으로 도입할 때 require를 사용해야 하는 이유는 무엇입니까? 다음 기사에서는 당신이 모르는 것에 대해 이야기할 것입니다. 도움이 되길 바랍니다!
Vue를 사용해 본 친구들은 면접관으로부터 다음과 같은 질문을 받았을 것입니다. vue에서 이미지를 동적으로 소개하기 위해 require를 사용해야 하는 이유는 무엇입니까? [관련 추천 : vuejs 영상 튜토리얼, 웹 프론트엔드 개발]
어떤 친구들은 비웃을지도 모릅니다: 하, 그렇군요, 동적으로 추가된 src는 정적 리소스로 취급되어 컴파일되지 않기 때문에 need to be require를 추가하면 거꾸로도 외울 수 있어요...
으음... 언뜻 보면 말이 되는 것 같지만, 자세히 보면 이 문장이 정확히 무슨 뜻인가요? 위 답변에 대해 다음과 같은 질문을 가지지 않을 수 없습니다.
정적 리소스란 무엇입니까?
동적으로 추가된 src를 정적 리소스로 간주하는 이유는 무엇인가요?
컴파일되지 않았다는 것은 무엇을 의미합니까?
require를 추가하면 리소스를 올바르게 가져올 수 있는 이유는 require를 추가하면 컴파일이 가능하기 때문인가요?
마지막 질문을 했을 때 위의 답변이 뭔가를 말하고 있는 것 같았는데 아무 말도 하지 않는 것 같았어요... 독자 여러분도 위의 질문을 갖고 계시다면 설명해 드리겠습니다. one by one
1. 정적 리소스란 무엇인가요? 정적 리소스에 이어 동적 리소스도 있습니다. 먼저 인터넷의 유명 인사들이 이를 어떻게 설명하는지 살펴보겠습니다.
정적 리소스: 일반적으로 클라이언트는 웹 서버에 요청을 보냅니다. 웹 서버는 메모리에서 해당 파일을 검색하여 클라이언트에 반환합니다. 동적 리소스: 일반적으로 클라이언트가 요청한 동적 리소스의 경우 요청이 먼저 웹 컨테이너로 전달되고, 웹 컨테이너는 데이터베이스에 연결되어 데이터를 처리한 후 콘텐츠가 웹으로 전달됩니다. 서버에 전달되고 웹 서버는 구문 분석 및 렌더링 처리를 위해 이를 클라이언트에 반환합니다.사실 위의 요약은 매우 명확합니다. Vue 프로젝트의 관점에서 보면 다음과 같이 간단하게 이해할 수 있습니다.정적 리소스는 프로젝트에 직접 저장되는 리소스입니다. 이러한 리소스는 리소스를 얻기 위해 특별한 요청을 보낼 필요가 없습니다
. 예를 들어 자산 디렉토리 아래의 사진, 비디오, 오디오, 글꼴 파일, CSS 스타일 시트 등입니다.
동적 리소스는 요청을 보내서 얻어야 하는 리소스입니다. 예를 들어, Taobao를 탐색할 때 동적 리소스라고 할 수 있는 특별 요청을 보내 다양한 제품 정보를 얻습니다.
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。
在回答这个问题之前,容我先对问题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="..." alt="Vue에 이미지를 동적으로 도입하기 위해 require를 사용해야 하는 이유는 무엇입니까? 분석 및 공유" >
、background: url(...)
和 CSS@import
的资源 URL 都会被解析为一个模块依赖。例如,
url(./Vue에 이미지를 동적으로 도입하기 위해 require를 사용해야 하는 이유는 무엇입니까? 분석 및 공유)
会被翻译为require('./Vue에 이미지를 동적으로 도입하기 위해 require를 사용해야 하는 이유는 무엇입니까? 분석 및 공유')
,而:<img src="./Vue에 이미지를 동적으로 도입하기 위해 require를 사용해야 하는 이유는 무엇입니까? 분석 및 공유" alt="Vue에 이미지를 동적으로 도입하기 위해 require를 사용해야 하는 이유는 무엇입니까? 분석 및 공유" >로그인 후 복사将会被编译到:
h('img', { attrs: { src: require('./Vue에 이미지를 동적으로 도입하기 위해 require를 사용해야 하는 이유는 무엇입니까? 분석 및 공유') }})로그인 후 복사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,既可以减少请求,也不会影响到页面的加载性能。
(学习视频分享:编程基础视频)
위 내용은 Vue에 이미지를 동적으로 도입하기 위해 require를 사용해야 하는 이유는 무엇입니까? 분석 및 공유의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!