Webpack5 新特性
在使用webpack的时候,我们常常会做一些优化,比如:
构建速度优化
代码体积优化
持久化缓存优化
Module Federation
到了webpack5,这些优化措施都变得更加的简单和效果显著了
构建速度优化
在webpack4中,为了让我们的构建速度更快,我们通常需要借助一些插件或一些额外的配置来达到目的. eg:
cache-loader
,针对一些耗时的工作进行缓存。比如缓存babel-loader
的工作terser-webpack-plugin
或uglifyjs-webpack-plugin
的 cache 以及 parallel。(默认开启)
比如我们会借助 cache-loader
去对我们构建过程中消耗性能比较大的部分进行缓存,缓存会存放到硬盘中 node_modules/.cache/cache-loader
,缓存的读取和存储是会消耗性能的,所以只推荐用在性能开销大的地方
// 对babel-loader的工作进行缓存
module.exports = {
module: {
rules: [
{
test: /\.js$/,
use: ['cache-loader', 'babel-loader'],
include: path.resolve('src'),
},
],
},
};
terserPlugin 继承自 uglifyjsPlugin,我们可以开启插件的 cache
以及 parallel
特性来加快压缩。( terserPlugin 是 Webpack 推荐及内置的压缩插件,cache
与 parallel
默认为开启状态)缓存路径在 node_modules/.cache/terser-webpack-plugin
optimization: {
minimizer: [
new TerserPlugin({
cache: true, // 开启该插件的缓存,默认缓存到node_modules/.cache中
parallel: true, // 开启“多线程”,提高压缩效率
exclude: /node_modules/
})
],
},
到了 Webpack5 ,可以通过 cache
特性来将 Webpack 工作缓存到硬盘中。存放的路径为 node_modules/.cache/webpack
开发环境默认值为
cache.type = "memory"
生产环境可手动设为
cache.type = "filesystem"
module.exports = {
//...
cache: {
type: 'filesystem',
version: 'your_version'
}
}
持久化缓存的优化
在日常开发中我们会尽量减少文件 hash
发生变化的情况,以最大化的利用缓存,节省流量。这就是我们常说的“优化持久化缓存”。首先最简单的措施就是使用 contenthash
来作为文件哈希后缀,只有当文件内容发生变化的时候,哈希才会发生改变。但是这样并不够。我们还是会遇到这样的问题:
当我们新增一个模块时:
// 在入口文件index.js新增了模块demo
// ...
import {a} from './demo'
console.log(a);
// ...
所有文件的哈希后缀都发生了改变,不符合期望,vender~xxx.js的hash不应发生变化
继续当我们新增一个入口的时候:
entry: {
index: ['./src/index.js'],
index2: ['./src/index2.js']
},
同样的所有文件的哈希后缀都发生了改变,不符合期望,原有文件hash不应发生变化。
问题原因
在 Webpack4 中,chunkId 与 moduleId
都是自增 id
。也就是只要我们新增一个模块,那么代码中 module
的数量就会发生变化,从而导致 moduleId
发生变化,于是文件内容就发生了变化。chunkId 也是如此,新增一个入口的时候,
chunk 数量的变化造成了
chunkId `的变化,导致了文件内容变化
解决方法
Webpack4可以通过设置 optimization.moduleIds = 'hashed'
与 optimization.namedChunks=true
来解决这写问题,但都有性能损耗等副作用
optimization: {
moduleIds: 'hashed',
namedChunks: true,
// ...
}
而 Webpack5 在 production
模式下 optimization.chunkIds
和 optimization.moduleIds
默认会设为 'deterministic'
,Webpack会采用新的算法来计算确定性的 chunkId 和 moduleId。默认即可避免上述情况发生。
包代码体积的优化
为了让我们的打出来的包体积更加小,颗粒度更加明确。我们经常会用到 Webpack 的代码分割 splitchunk 以及 tree shaking。在 Webpack5 中,这两者也得到了优化与加强
SplitChunks
SplitChunks 配置示例:
splitChunks: {
chunks: 'all',
minSize: {
javascript: 30000,
style: 50000,
}
},
// 默认配置
module.exports = {
//...
// https://github.com/webpack/changelog-v5#changes-to-the-configuration
// https://webpack.js.org/plugins/split-chunks-plugin/
optimization: {
splitChunks: {
chunks: 'async', // 只对异步加载的模块进行处理
minSize: {
javascript: 30000, // 模块要大于30kb才会进行提取
style: 50000, // 模块要大于50kb才会进行提取
},
minRemainingSize: 0, // 代码分割后,文件size必须大于该值 (v5 新增)
maxSize: 0,
minChunks: 1, // 被提取的模块必须被引用1次
maxAsyncRequests: 6, // 异步加载代码时同时进行的最大请求数不得超过6个
maxInitialRequests: 4, // 入口文件加载时最大同时请求数不得超过4个
automaticNameDelimiter: '~', // 模块文件名称前缀
cacheGroups: {
// 分组,可继承或覆盖外层配置
// 将来自node_modules的模块提取到一个公共文件中 (又v4的vendors改名而来)
defaultVendors: {
test: /[\\/]node_modules[\\/]/,
priority: -10
},
// 其他不是node_modules中的模块,如果有被引用不少于2次,那么也提取出来
default: {
minChunks: 2,
priority: -20,
reuseExistingChunk: true
}
}
}
}
};
Tree Shaking
同时tree shaking也得到了加强,可以覆盖更多场景
Nested tree-shaking
Inner-module tree-shaking
Node.js Polyfills
webpack5之前,Webpack 会自动的帮我们项目引入 Node 全局模块 polyfill。我们可以通过 node 配置
// false: 不提供任何方法(可能会造成bug),'empty': 引入空模块, 'mock': 引入一个mock模块,但功能很少
module.exports = {
// ...
node: {
console: false,
global: false,
process: false,
// ...
}
}
但是 Webpack 团队认为,现在大多数工具包多是为前端用途而编写的,所以不再自动引入 polyfill。我们需要自行判断是否需要引入 polyfill,当我们用Webpack5 打包的时候,Webpack会给我们类似如下的提示:
// 在项目中我使用到了 crypto 模块,webpack5会询问是否引入对应的 polyfill。
Module not found: Error: Can't resolve 'crypto' in '/Users/xxx/Documents/private-project/webpack/ac_repair_mobile_webpack_5/node_modules/sshpk/lib/formats'
BREAKING CHANGE: webpack < 5 used to include polyfills for node.js core modules by default.
This is no longer the case. Verify if you need these module and configure a polyfill for it.
If you want to include a polyfill, you need to:
- add an alias 'resolve.alias: { "crypto": "crypto-browserify" }'
- install 'crypto-browserify'
If you don't want to include a polyfill, you can use an empty module like this:
resolve.alias: { "crypto": false }
Webpack5 中,增加了 resolve.alias
配置项来告诉 Webpack 是否需要引入对应 polyfill。node 配置项也做了调整
module.exports = {
// ...
resolve: {
alias: {
crypto: 'crypto-browserify',
// ..
}
},
node: {
// https://webpack.js.org/configuration/node/#root
// 只能配置这三个
global: false,
__filename: false,
__dirname: false,
}
}
也就是说到了 Webpack5,我们需要清楚自己的项目需要引入哪些 node polyfill。更加了配置的门槛,但是减少了代码的体积。
webpack5 中将 path
、crypto
、http
、stream
、zlib
、vm
的 node polyfill取消后效果如下图:
Module Federation
模块联邦制,使 JavaScript 应用得以从另一个 JavaScript 应用中动态地加载代码 —— 同时共享依赖。项目分为Host(消费者),remote(被消费者)。功能实现主要依靠 ModuleFederationPlugin 插件
new ModuleFederationPlugin({
name: '', // 名称,唯一id
library: {}, // 以什么形式暴露,比如umd
filename: '', // 输出的入口文件名称
exposes: {}, // 要输出的组件或方法
shared: [] // 要共享的依赖
})
比如 app1, 输出log方法
new ModuleFederationPlugin({
name: 'app1',
library: {type: 'var', name: 'app1'},
filename: 'appOneEntry.js',
exposes: {
'./log': './util/logSomething'
},
shared: []
})
app2 使用 app1 中的 logSomething
方法
new ModuleFederationPlugin({
name: "app2",
remotes: {
app1: 'app1@http://127.0.0.1:8887/demo-federation-1/dist/appOneEntry.js'
},
shared: []
})