共计 3905 个字符,预计需要花费 10 分钟才能阅读完成。
简介
读了《深入浅出 webpack》总结一下常用的 webpack 的构建优化策略,可通过以下手段来提升项目构建时的速度
更精准的 loader 规则
将 loader 规则写清楚
仅让需要处理的文件,进入 loader
处理环节,如下
rules: [{
// 正则尽量准确
test: /\.js$/,
// 使用缓存,缓存后在文件未改变时编译会更快(缓存查找原理见补充 1)
use: ['babel-loader?cacheDirectory'],
// 指定需要处理的目录
include: path.resolve(__dirname, 'src')
// 理论上只有 include 就够了,但是某些情况需要排除文件的时候可以用这个,排除不需要处理文件
// exclude: []}]
更精准的查找目录
将查找路径设置精确
理论上我们项目的第三方依赖均应在自己的工程的 node_modules
下,所以我们可以设置查找目录,减少 node 的默认查找(默认查找方式见补充 2)
module.exports = {
resolve: {
// 指定当前目录下的 node_modules 目录
modules: [path.resolve(__dirname, 'node_modules')]
}
}
更精准的扩展名
数量更多类型的文件尽量放在前面
平时写代码,我们都习惯直接写文件名,而不去写扩展名,那么解析则按照下面属性进行解析
module.exports = {extensions: ['.js', '.jsx', '.ts', '.tsx'],
}
默认值
extensions: [".js", ".json"]
使用动态链接库预编译大模块
使用动态链接库,提前编译大模块
原理请见补充 3
新建一个文件webpack_dll.config.js
, 内容如下
const path = require('path');
const webpack = require('webpack');
// 复用的大模块放在这里,这样每次都不需要重新编译了
const vendors = [
'react',
'react-dom',
'lodash'
];
module.exports = {
mode: 'development',
output: {path: path.resolve(__dirname, './dist'),
filename: '[name].js',
library: '[name]',
},
entry: {vendors,},
plugins: [
new webpack.DllPlugin({path: path.resolve(__dirname, './dist/manifest.json'),
name: '[name]',
}),
],
};
执行 webpack --config webpack_dll.config.js
进行首次编译(如果更新版本需要再次编译)
然后在你的 webpack
配置文件中引入manifest.json
plugins: [
new webpack.DllReferencePlugin({manifest: require('./dist/manifest.json')
})
],
多进程处理文件
使用 HappyPack 同时处理多个 loader 编译任务
为了发挥多核 CPU 电脑的功能,利用 HappyPack
将任务分发给多个子进程并发执行
const path = require('path');
const HappyPack = require('happypack');
// 共享 5 个进程池
const happyThreadPool = HappyPack.ThreadPool({size: 5});
module.exports = {
entry: './src/index.js',
output: {
filename: 'bundle.js',
path: path.resolve(__dirname, 'dist'),
},
module: {// noParse: [/react\.production\.min\.js$/],
rules: [{
test: /\.js$/,
// 和下面插件 id 一直,happypack 才可以找到
use: ['happypack/loader?id=babel'],
include: path.resolve(__dirname, 'src')
}]
},
plugins: [
// 插件可以实例化多个
new HappyPack({
// 与上面对应
id: 'babel',
// 实际要使用的 loader
loaders: ['babel-loader?cacheDirectory'],
// 默认开启进程数
threads: 3,
// 是否允许 happyPack 打印日志
verbose: true,
// 共享进程数,如果你使用超过一个 happyplugin,官方建议共享进程池
threadPool: happyThreadPool
})
],
};
原理见补充 4
多进程压缩文件
使用 ParallelUglifyPlugin 多进程同时压缩文件
ParallelUglifyPlugin
是在 UglifyJS
基础上,增加了多进出处理的能力,加快了压缩速度
import ParallelUglifyPlugin from 'webpack-parallel-uglify-plugin';
module.exports = {
plugins: [
new ParallelUglifyPlugin({
test,
include,
exclude,
cacheDir,
workerCount,
sourceMap,
uglifyJS: { },
uglifyES: {}}),
],
};
减少监听文件
减少监听文件
原理见补充 5
当我们使用 webpack
的watch
功能进行文件监听时,更好的方式是控制监听目录,如下,排除 node_modules
减少对该目录监听,减少编译所需要循环的文件,提高检查速度
module.export = {
watchOptions: {ignored: /node_modules/}
}
其他没那么重要的优化
更精准的 mainFields
默认的这个值查找方式见官网点击此处
看了下 react
和lodash
,只有一个main
,目前来看使用 es6 看来还不普遍,所以这个值目前可能不太重要
module.exports = {
resolve: {mainFields: ['main']
}
}
第三方库映射
为什么这个不重要,我发现 react
直接导出的 index.js
则是根据环境判断使用哪份代码,目测来看并不需要进行循环依赖的处理
通过依赖,则可以直接使用打包后代码,而不需 webpack 去循环依赖
resolve: {mainFields: ["main"],
alias: {'react': path.resolve(__dirname, './node_modules/react/cjs/react.production.min.js')
}
}
不使用 inline 模式的 devServer
原理见补充 6
默认情况下,应用程序启用内联模式 (inline mode)。这意味着一段处理实时重载的脚本被插入到你的包(bundle) 中,并且构建消息将会出现在浏览器控制台。
当使用 inline
模式时,devServer
会向每个 Chunk
中注入一段重载的脚本代码,但是其实一个页面只需要一次,所以当 Chunk
过多时,可以将 inline
设置为false
module.export = {
devServer: {inline: false}
}
补充
补充 1 -cacheDirectory 原理
当有设置 cacheDirectory
时,指定的目录将用来缓存 loader
的执行结果。之后的 webpack 构建,将会尝试读取缓存,来避免在每次执行时,可能产生的、高性能消耗的 Babel
重新编译过程。如果设置了一个空值 (loader: ‘babel-loader?cacheDirectory’) 或者 true (loader: babel-loader?cacheDirectory=true),loader 将使用默认的缓存目录 node_modules/.cache/babel-loader,如果在任何根目录下都没有找到 node_modules 目录,将会降级回退到操作系统默认的临时文件目录。
补充 2 -node 的默认查找方式
- 查找当前目录下的
node_modules
目录,看是否有匹配项,如有,命中文件 - 寻找父目录的下的
node_modules
,如有,命中文件 - 按照这个规则一直往父目录搜索直到到根目录下的
node_modules
补充 3 - 动态链接库思想
大量项目中可以复用的模块只需要被编译一次,在之后的构建过程中被动态链接库包含的模块将不会重新编译,而是直接使用动态链接库中的代码。(注:如果升级依赖的模块版本,需要重新编译动态链接库)
补充 4 -HappyPack 原理
webpack
构建中,需要大量的 loader 转换操作,很耗时,由于 nodejs 是单线程的,如果想更好利用 cpu 的多核能力,可以开启多个进程,同时对文件进行处理;可以看到在配置文件中,我们每次将文件交给 happypack-loader
去处理,然后由 happypack
去调度来执行文件的处理(happypack
采用哪个 loaders
进行处理,是通过 id
知道的)
补充 5 - 文件监听原理
webpack
会从入口触发,将所有的依赖项放到一个 list 里边,然后每次修改文件内容,回去遍历整个 list 里边的文件,看是否有编辑时间的变化,如果有的话则进行编译
补充 6 - 自动刷新原理
- 向要开发的网页中注入代理客户端代码,通过代理客户端去刷新整个页面(默认)
- 将要开发的网页放进一个 iframe,通过刷新 iframe 去看刷新效果