本指南旨在帮助你在直接使用 webpack 时迁移到 webpack 5。如果你正在使用更高级别的工具来运行 webpack,请参考该工具的迁移说明。
Webpack 5 至少需要 Node.js 10.13.0 (LTS),因此,如果你仍在运行较旧版本,请确保升级你的 Node.js。
将 webpack
4 升级到最新可用版本。
使用 webpack >= 4 时,升级到最新 webpack 4 版本不需要额外的指导。
如果您使用的 webpack 版本低于 4,请参阅 webpack 4 迁移指南。
将 webpack-cli
升级到最新可用版本(如果使用)
将所有使用的插件和加载器升级到最新可用版本
某些插件和加载器可能具有必须使用才能与 webpack 5 兼容的测试版。升级时务必阅读每个单独插件/加载器的发行说明,因为最新版本可能仅支持 webpack 5,并且在 v4 中会失败。在这种情况下,建议更新到支持 webpack 4 的最新版本。
由于 webpack
、webpack-cli
、插件和加载器的版本升级,可能会出现新的错误或警告。在构建过程中注意弃用警告。
您可以通过这种方式调用 webpack
以获取弃用警告的堆栈跟踪,以找出哪些插件和加载器负责。
node --trace-deprecation node_modules/webpack/bin/webpack.js
由于 webpack 5 删除了所有已弃用的功能,因此请确保在构建期间没有 webpack 弃用警告才能继续。
mode
将 mode
设置为 production
或 development
以确保设置相应的默认值。
将以下选项更新到其新版本(如果使用)
optimization.hashedModuleIds: true
→ optimization.moduleIds: 'hashed'
optimization.namedChunks: true
→ optimization.chunkIds: 'named'
optimization.namedModules: true
→ optimization.moduleIds: 'named'
NamedModulesPlugin
→ optimization.moduleIds: 'named'
NamedChunksPlugin
→ optimization.chunkIds: 'named'
HashedModuleIdsPlugin
→ optimization.moduleIds: 'hashed'
optimization.noEmitOnErrors: false
→ optimization.emitOnErrors: true
optimization.occurrenceOrder: true
→ optimization: { chunkIds: 'total-size', moduleIds: 'size' }
optimization.splitChunks.cacheGroups.vendors
→ optimization.splitChunks.cacheGroups.defaultVendors
optimization.splitChunks.cacheGroups.test(module, chunks)
→ optimization.splitChunks.cacheGroups.test(module, { chunkGraph, moduleGraph })
Compilation.entries
→ Compilation.entryDependencies
serve
→ serve
已被弃用,建议使用 DevServer
Rule.query
(自 v3 起已弃用)→ Rule.options
/UseEntry.options
Rule.loaders
→ Rule.use
尝试在 webpack 4 配置中设置以下选项,并检查构建是否仍然正常工作。
module.exports = {
// ...
node: {
Buffer: false,
process: false,
},
};
在为 webpack 5 升级配置时,必须再次删除这些选项。
现在,让我们将 webpack 升级到版本 5
npm:npm install webpack@latest
Yarn:yarn add webpack@latest
如果你无法在“升级 webpack 4 及其插件/加载器”步骤中将某些插件/加载器升级到最新版本,请不要忘记现在升级它们。
考虑从 webpack 配置中删除 optimization.moduleIds
和 optimization.chunkIds
。默认值可能会更好,因为它们支持 生产模式
中的长期缓存和 开发
模式 中的调试。
在 webpack 配置中使用 [hash]
占位符时,请考虑将其更改为 [contenthash]
。它并不相同,但已被证明更有效。
如果你正在使用 Yarn 的 PnP 和 pnp-webpack-plugin
,我们有一个好消息:它现在默认受支持。你必须从配置中将其删除。
如果你正在将 IgnorePlugin
与正则表达式一起用作参数,它现在需要一个 options
对象:new IgnorePlugin({ resourceRegExp: /regExp/ })
。
如果你正在使用类似 node.fs: 'empty'
的内容,请将其替换为 resolve.fallback.fs: false
。
如果你在 webpack Node.js API 中使用 watch: true
,请将其删除。无需设置它,因为它是由你调用的编译器方法指示的,对于 watch()
来说是 true
,对于 run()
来说是 false
。
如果你有使用 raw-loader
、url-loader
或 file-loader
加载资产的已定义 rules
,请改用 资产模块,因为它们将在不久的将来被弃用。
如果你将 target
设置为函数,请将其更新为 false
,并在 plugins
选项中应用该函数。请参阅以下示例
// for webpack 4
{
target: WebExtensionTarget(nodeConfig)
}
// for webpack 5
{
target: false,
plugins: [
WebExtensionTarget(nodeConfig)
]
}
如果您定义了 output.library 或 output.libraryTarget,请更改属性名称:(output.libraryTarget -> output.library.type,output.library -> output.library.name)。示例
// for webpack 4
{
output: {
library: 'MyLibrary',
libraryTarget: 'commonjs2'
}
}
// for webpack 5
{
output: {
library: {
name: 'MyLibrary',
type: 'commonjs2'
}
}
}
如果您正在通过导入使用 WebAssembly,则应遵循以下两步流程
experiments.syncWebAssembly: true
启用已弃用的规范,以获得与 webpack 4 中相同的行为。experiments
值更改为 experiments: { asyncWebAssembly: true }
以使用最新的 WASM 集成规范。重新考虑 optimization.splitChunks
optimization.splitChunks: { chunks: 'all' }
。name: false
并用 idHint: string | function
替换 name: string | function
。optimization.splitChunks.cacheGroups: { default: false, vendors: false }
来关闭默认值。我们不建议这样做,但如果您真的想在 webpack 5 中获得相同的效果:optimization.splitChunks.cacheGroups: { default: false, defaultVendors: false }
。考虑删除默认值
entry: './src/index.js'
:您可以省略它,这是默认值。output.path: path.resolve(__dirname, 'dist')
:您可以省略它,这是默认值。output.filename: '[name].js'
:您可以省略它,这是默认值。如果您为您的项目启用了 browserslist,webpack 5 将重复使用您的 browserslist
配置来决定为运行时代码发出哪种代码样式。
请确保
target
设置为 browserslist
或删除 target
,让 webpack 自动为您设置 browserslist
。IE 11
添加到您的 browserslist 配置中。如果没有 browserslist
,webpack 的运行时代码将使用 ES2015 语法(例如箭头函数)来构建更小的包。因此,你需要将 target: ['web', 'es5']
设置为对浏览器(如 IE11)使用 ES5 语法,因为它们不支持 ES2015 语法。
对于 Node.js,构建在 target
选项中包含受支持的 Node.js 版本,并且 webpack 将自动找出支持哪种语法,例如 target: 'node8.6'
。
/* webpackChunkName: '...' */
确保理解意图
webpackChunkName
,Webpack 5 也将在 development
模式下自动分配有用的文件名。新规范不支持此操作,你将收到警告。请不要
import { version } from './package.json';
console.log(version);
使用
import pkg from './package.json';
console.log(pkg.version);
const compiler = webpack(...);
时,请确保在使用后关闭编译器:compiler.close(callback);
。webpack(..., callback)
形式。请务必仔细阅读构建错误/警告。如果没有相应的建议,请创建一个问题,我们会尝试解决它。
重复以下步骤,直到至少解决 3 级或 4 级
1 级:架构验证失败。
配置选项已更改。应出现一个验证错误,其中包含 BREAKING CHANGE:
注释,或提示应使用哪个选项。
2 级:Webpack 退出时出现错误。
错误消息应告诉你需要更改的内容。
3 级:构建错误。
错误消息应包含 BREAKING CHANGE:
注释。
第 4 级:生成警告。
警告消息应告知您哪些方面可以改进。
第 5 级:运行时错误。
这很棘手。您可能需要调试来查找问题。此处难以提供一般性建议。但我们确实在下面列出了一些有关运行时错误的常见建议
process
未定义。exports
或 imports
package.json 字段,根据环境使用不同的代码。browser
字段来支持较旧的打包器。typeof process
检查包装代码块。请注意,这会对包大小产生负面影响。process.env.VARIABLE
中的环境变量吗?您需要使用 DefinePlugin
或 EnvironmentPlugin
在配置中定义这些变量。VARIABLE
,并确保也检查 typeof VARIABLE !== 'undefined'
。process.env
属于 Node.js 特有,应避免在前端代码中使用。auto
的 URL 的 404 错误output.publicPath: "auto"
实现的新默认自动 publicPath
做好准备output.publicPath: ""
。第 6 级:弃用警告。
您可能会收到许多弃用警告。这并非直接问题。插件需要时间来赶上核心更改。请向插件报告这些弃用。这些弃用只是警告,并且生成仍将继续进行,仅有轻微缺点(如性能下降)。
--no-deprecation
标志运行节点来隐藏弃用警告,例如:node --no-deprecation node_modules/webpack/bin/webpack.js
。这应仅作为临时解决办法。第 7 级:性能问题。
通常,webpack 5 的性能应该会有所提高,但也有少数情况下性能会变差。
以下是可以采取的一些措施来改善这种情况
--profile --progress
现在会显示一个简单的性能分析node --inspect-brk node_modules/webpack/bin/webpack.js
+ chrome://inspect
/ edge://inspect
(请参见分析器选项卡)。--no-turbo-inlining
标记,在某些情况下可获得更好的堆栈跟踪。module.unsafeCache: true
devtool
选项,了解不同选项的比较。请发推文表示您已成功迁移到 webpack 5。 发推文
创建 问题,并告诉我们您在迁移期间遇到的问题。
请打开 Pull Request,帮助使用本指南的其他人。
对于有兴趣的人,此处列出了 webpack 内部更改,例如:添加类型、重构代码和方法重命名。但它们并非作为常见用例迁移的一部分。
Module.nameForCondition
、Module.updateCacheModule
和 Module.chunkCondition
不再是可选的。Webpack 5 内置 this.getOptions
方法,可在加载器上下文中使用。对于之前使用 schema-utils 中首选的 getOptions
方法的加载器来说,这是一个重大更改
this.getOptions
自 webpack 5 起可用?{arg:true}
→ ?{"arg":true}
。在各自的加载器文档中,应将使用 JSON5 视为已弃用,并记录为 JSON 的替代方案。loader-utils
具有解析查询字符串的特定行为(true
、false
和 null
不会解析为 string
,而是解析为原始值)。对于新的内置 this.getOptions
方法不再是这种情况,它使用本机 querystring
解析(随 Node.js 一起提供)。在使用 this.getOptions
方法获取选项后,仍然可以在加载器代码中为这些情况添加自定义行为。this.getOptions
方法,架构参数是可选的,但我们强烈建议为加载器的选项添加架构验证。架构中的 title
字段可用于自定义验证错误消息,例如 "title": "My Loader ooooptions"
将以这种方式显示错误:Invalid ooooptions object. My Loader has been initialised using an ooooptions object that does not match the API schema. - ooooptions.foo.bar.baz should be a string.