Webpack 4教程 - 第八部分 使用prefetch和preload进行动态加载
转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具、解决方案和服务,赋能开发者。
原文出处:https://wanago.io/2018/08/13/webpack-4-course-part-seven-decreasing-the-bundle-size-with-tree-shaking/
在本系列的第一篇文章中,我们讨论了导入(import)和导出(export)。这次我们深入介绍动态导入(dynamic import),它值得专门用一篇文章来介绍。我们会介绍动态导入是什么以及如何使用它们。开始吧!
在过去,ECMAScript模块是完全静态的。你必须在运行代码之前指明想要导入和导出的东西。随着动态导入提案的出现,我们有了额外的选择,即动态地导入模块。现在它进行到了TC39流程的第三个阶段。有了它,你就可以添加动态导入模块了。使用它时,你可能会根据用户及其操作行为的做相应处理。比如,你有一个单页应用,只有当用户决定打开它的子页面时才加载特定代码。这样可以大幅节省应用的初始加载时间。
使用动态导入
动态导入操作符是作为函数使用的。它接受一个字符串参数,返回一个Promise。当模块加载好后,这个Promise被resolve。
如果你想了解更多关于Promise的内容,可查看以实现一个排序算法为例解释Promise和回调函数。
document.addEventListener("DOMContentLoaded", () => {
const button = document.querySelector('#divideButton');
button.addEventListener('click', () => {
import('./utilities/divide')
.then(divideModule => {
console.log(divideModule.divide(6, 3)); // 2
})
});
});
在浏览器的开发者工具,如果打开Network标签,你可以看到,模块开始下载的发生在点击按钮之后,而不是在此之前。值得注意的是,如果再次点击按钮,包含了拆分后的模块文件不会再次被下载。
在Webpack中使用动态导入,会新增一个chunk,我们视作异步chunk。
异步chunk在教程的第四部分-使用SplitChunksPlugin分离代码中有介绍。
像这样的chunk会被打包进单独的文件。当使用表达式创建指向其文件的路径时,你需要小心。考虑如下例子:
let fileName = ''; document.addEventListener("DOMContentLoaded", () => {
const button = document.querySelector('#divideButton');
fileName = 'divide';
button.addEventListener('click', () => {
import(`./utilities/${fileName}`)
.then(divideModule => {
console.log(divideModule.divide(6, 3)); // 2
})
});
});
以上代码在你的项目中被打包过后,你会发现Webpack在utilities文件夹下为每个模块单独创建了异步chunk。这是因为Webpack不能在编译时知道哪些模块需要被导入。
你还需要知道像import(pathToFile)这样的完全的动态声明是不起作用的,因为Webpack至少需要一部分文件路径信息。这是因为pathToFile可以是你工程中任何文件的路径,而Webpack会为每个模块在给定的文件夹中创建异步chunk。你可以自定义此行为,我们下面就会这么做。
使用在Webpack中使用魔法注释
导入模块的规范不允许你在导入时使用除了文件名以外的参数。幸运的是,有了Webpack,你可以利用所谓的**魔法注释(magic comments)**来使用附加参数。
webpackInclude 和 webpackExclude
在之前的小节,我们提到Webpack会为每个模块在我们给定的文件夹中创建异步chunk。虽然这是默认行为,但它可以修改。
其中一种方法是使用webpackExclude,它是一个正则表达式,用以匹配潜在的可被导入的文件。任何匹配到的文件都不会被打包进来。
import(
`./utilities/${fileName}`
/* webpackExclude: /subtract.js$/ */
)
以上代码表示,文件 subtract.js 文件不会被打包进来,即使它在 utilities 目录下。
与之相反的一个参数叫做webpackInclude。使用它时,只有匹配了正则表达式的模块会被打包。
webpackMode
webpackMode属性定义了resolve动态模块时的模式。支持以下模式:
lazy
这是默认模式。它为每个动态导入的模块创建异步chunk。
lazy-once
使用它,会为满足导入条件的所有模块创建单一的异步chunk。
import(
`./utilities/${fileName}`
/* webpackMode: "lazy-once" */
)
.then(divideModule => {
console.log(divideModule.divide(6, 3)); // 2
})
以上代码表示,Webpack会为所有 utilities 目录下的所有模块共同创建一个异步chunk。它会导致用户以一个文件下载所有的模块。
eager
此模式会阻止Webpack生成额外的chunk。所有导入的模块被包含在当前chunk,所以不需要再发额外的网络请求。它仍然返回一个Promise,但它被自动resolve。使用eager模式的动态导入与静态导入的区别在于,整个模块只有当**import()**掉用之后才执行。
weak
彻底阻止额外的网络请求。只有当该模块已在其他地方被加载过了之后,Promise才被resolve,否则直接被reject。
webpackChunkName
它是新chunk的名字,可以和[index]、[request]变量一起使用。
[index]在当前动态导入声明中表示文件的索引。另一方面,[request]表示导入文件的动态部分。
import(
`./utilities/${fileName}`
/* webpackChunkName: "utilities-[index]-[request]" */
)
以上代码可能生成例如 utilities-0-divide.js 这样的文件名。
请注意,如果在某些情况下,确定只有一个异步chunk(比如本来就没有动态生成路径,或者使用了lazy-once模式),[index]和[request]就不会被使用了。
使用预先拉取和预先加载提升性能
Webpack 4.6.0为我们提供了预先拉取(prefetching)和预先加载(preloading)的功能。使用这些声明可以修改浏览器处理异步chunk的方式。
预先拉取
使用预先拉取,你表示该模块可能以后会用到。浏览器会在空闲时间下载该模块,且下载是发生在父级chunk加载完成之后。
import(
`./utilities/divide`
/* webpackPrefetch: true */
/* webpackChunkName: "utilities" */
)
以上的导入会让<link rel="prefetch" as="script" href="utilities.js">被添加至页面的头部。因此浏览器会在空闲时间预先拉取该文件。
预先加载
在资源上添加预先加载的注释,你指明该模块需要立即被使用。异步chunk会和父级chunk并行加载。如果父级chunk先下载好,页面就已可显示了,同时等待异步chunk的下载。这能大幅提升性能。
import(
`./utilities/divide`
/* webpackPreload: true */
/* webpackChunkName: "utilities" */
)
以上代码的效果是让<link rel="preload" as="script" href="utilities.js">起作用。不当地使用wepbackPreload会损害性能,所以使用的时候要小心。
总结
这次我们学习了如何使用动态导入提升应用的性能。它们能显著减少页面的初次加载时间。使用可传入Webpack的额外参数,你可以更进一步地定制它,并且添加上对预先拉取和预先加载的支持。所有这些,都会优化你的用户体验,让你的网站更加灵动。
Webpack 4教程 - 第八部分 使用prefetch和preload进行动态加载的更多相关文章
- 【Silverlight】Bing Maps学习系列(八):使用Bing Maps Silverlight Control加载自己部署的Google Maps
[Silverlight]Bing Maps学习系列(八):使用Bing Maps Silverlight Control加载自己部署的Google Maps 上个月微软必应地图(Bing Maps) ...
- webpack动态加载打包chunk命名
最近,遇到复杂h5页面开发,为了优化H5首屏加载速度,想到使用按需加载的方式,减少首次加载的JavaScript文件体积,于是将处理过程在这里记录一下,涉及到的主要是以下三点: 使用Webpack如何 ...
- webpack分包:vue单页面解决分包【减少首屏加载时间】--按需加载[路由懒加载]
1.使用webpack中的syntax-dynamic-import 插件 npm install --save-dev babel-plugin-syntax-dynamic-import 2.配置 ...
- [ionic开源项目教程] - 第7讲 实现下拉刷新上拉加载ion-refresher和ion-infinite-scroll
第7讲 实现下拉刷新上拉加载ion-refresher和ion-infinite-scroll 1.将tab1.html的代码改为如下: <ion-content> <ion-ref ...
- Android简易实战教程--第三十二话《使用Lrucache和NetworkImageView加载图片》
转载本专栏每一篇博客请注明转载出处地址,尊重原创.此博客转载链接地址:小杨的博客 http://blog.csdn.net/qq_32059827/article/details/5279131 ...
- 安卓TV开发(八) 移动智能终端多媒体爬虫技术 获取加载网页视频源
转载请标明出处:http://blog.csdn.net/sk719887916/article/details/40049137,作者:skay 从上一篇学习中,学习了多媒体技术中的怎么去用josu ...
- Vue(二十八)el-cascader 动态加载 - 省市区组件
1.后台接口为点击加载下一级 ,传省市区id <template> <el-cascader v-model="selectedOptions" placehol ...
- vue问题八:动态加载loading、浏览器图标
全局动态加载loading: 参考文档:https://blog.csdn.net/qq_34825875/article/details/79570301 在main.js或者index.js引用 ...
- webpack散记---代码分割 和 懒加载
webpack methods ES 2015 Loader spec (1)webpack methods方法 require.ensure //可以动态加载依赖 []:dependencies / ...
随机推荐
- vc中调用Com组件的所有方法详解
首先,对于Com组件的入门学习,可以看一下<Windows程序设计技术基础——MFC与.NET> 任哲编著的21世纪重点大学规划教材那本书,适合入门(虽然不一定会使用),了解些基础原理. ...
- BZOJ 4199 [Noi2015]品酒大会:后缀数组 + 并查集
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=4199 题意: 给你一个长度为n的字符串s,和一个长为n的数组v. 对于每个整数r∈[0,n ...
- mysql 使用过程中出现问题
1. mysql_front连接报错,sql执行错误#3167的解决方案 提示:The 'INFORMATION_SCHEMA.SESSION_VARIABLES' feature is disabl ...
- 分享知识-快乐自己:Liunx 安装MySQL
第一步: 1):下载mysql安装包:这里选择下载版本 5.6.33,通用版,linux下64位 http://dev.mysql.com/get/Downloads/MySQL-5.6/mysql- ...
- fastjson转对象的一些属性设置
<bean class="com.alibaba.fastjson.support.spring.FastJsonHttpMessageConverter"> < ...
- GeoServer基础教程(三):部署发布Shapefile地图数据
转载:http://blog.csdn.net/mingzai624. 这是GeoServer官方网站提供的一份Shapefile测试数据 nyc_roads.zip ,包含了部分纽约的道路信息,我们 ...
- Linux-MySQL主从配置
1. MySQL主从原理以及应用场景MySQL的Replication原理非常简单,总结一下:每个从仅可以设置一个主.主在执行sql之后,记录二进制log文件(bin-log).从连接主,并从主获取b ...
- [原]NYOJ-括号匹配-2(java)
大学生程序代写 //http://acm.nyist.net/JudgeOnline/problem.php?pid=2 括号配对问题 时间限制:3000 ms | 内存限制:65535 KB ...
- Agc011_F Train Service Planning
先放题面,再放LHX巨佬题解 接着就是%%%.$orz.Oro.Or2.Otz.OTL.sto.rzo.Jto$.On_.○| ̄|_啊 模拟赛里直接把这道题刚掉了 一题升天·爆踩全场 这题思维跨越度已 ...
- 脚本手动执行正常,放cron中执行有问题的原因
问题原因: 1. crond服务没启动 2. 环境变量如 PATH LANG SHELL 等设置不对 3. 脚本中引用的文件地址是相对路径,而非绝对路径. 排查步骤: 以 check ...