最近将公司项目由webpack4升级到了webpack5,配置了webpack5的长效缓存后,二次构建速度直接提速了80%以上,同时打包体积也减少了,当然前提是要调研清楚坑多不多。

网上有一些做法是直接全部升级相关的包,而我是一个个包升级过来,如果有必要再升级,有的包没有升级(如babel-loader等等),项目本身用的大多也是较新的版本。

Webpack5 新特性

首先先要了解下升级 Webpack5能给我们带来什么好处,先看看 Webpack5的新特性

  • 通过持久化硬盘缓存能力来提升构建性能
  • 通过更好的算法来改进长期缓存(降低产物资源的缓存失效率)
  • 通过更好的 Tree Shaking 能力和代码的生成逻辑来优化产物的大小
  • 改善 web 平台的兼容性能力
  • 清除了内部结构中,在 Webpack4 没有重大更新而引入一些新特性时所遗留下来的一些奇怪的 state
  • 通过引入一些重大的变更为未来的一些特性做准备,使得能够长期的稳定在 Webpack5 版本上

对我来说最吸引人的还是构建性能的提升和更好的tree shaking 能力来优化产物的大小。

Webpack5 版本上为什么构建速度有了质的飞跃?

主要是因为:

webpack4是根据代码的结构生成chunkhash,添加了空白行或注释,会引起chunkhash的变化,webpack5是根据内容生成chunkhash,改了注释或者变量不会引起chunkhash的变化,浏览器可以继续使用缓存。

  • 优化了对缓存的使用效率。在webpack4 中,chunkId与moduleId都是自增id。只要我们新增一个模块,那么代码中module的数量就会发生变化,从而导致moduleId发生变化,于是文件内容就发生了变化。chunkId也是如此,新增一个入口的时候,chunk数量的变化造成了chunkId的变化,导致了文件内容变化。所以对实际未改变的chunk文件不能有效利用。webpack5采用新的算法来计算确定性的chunkId和moduleId。可以有效利用缓存。在production模式下,optimization.chunkIds和optimization.moduleIds默认会设为’deterministic’。
  • 新增了可以将缓存写入磁盘的配置项, 在命令行终止当前构建任务,再次启动构建时,可以复用上一次写入硬盘的缓存,加快构建过程。

简单了解上述后,简单记录下我在升级过程做的一些事情,要看完整的建议去官方升级文档查阅

直接升级依赖

以下一些库在我们项目直接升级到最新版本后是不需要做兼容的,本身配置不多,但是如果你项目插件写的配置项多就可能有需要做兼容的地方。

  • webpack webpack-cli
  • html-webpack-plugin
  • terser-webpack-plugin
  • mini-css-extract-plugin
  • style-loader、css-loader、postcss、postcss-loader(升级)

兼容

资源模块类型

在 webpack4及之前我们会用各种loader来处理一些资源,比如file-loader,url-loaderraw-loader等等,但webpack5内置了静态资源构建能力,不再需要安装这些额外的 loader,通过简单的配置就能实现静态资源的打包。

资源模块类型(asset module type),通过添加 4 种新的模块类型,来替换所有这些 loader:

  • asset/resource 发送一个单独的文件并导出 URL。之前通过使用 file-loader 实现。
  • asset/inline 导出一个资源的 data URI。之前通过使用 url-loader 实现。
  • asset/source 导出资源的源代码。之前通过使用 raw-loader 实现。
  • asset 在导出一个 data URI 和发送一个单独的文件之间自动选择。之前通过使用 url-loader,并且配置资源体积限制实现。

例子:

module: {
    rules: [
        {
            test: /\.(png|jpg|jpeg|gif)$/,
            type: `asset/resource`
        }
    ]
},

webpack-dev-server

需要注意的是:webpack-dev-server v4.0.0+ requires node >= v12.13.0

升级 webpack-dev-server 至 ^4(next) 版本,否则 HMR 会有异常

  • wbepack4 启动方式为:webpack-dev-server
  • webpack5 修改为:webpack server

    // v4
    devServer: {
      contentBase: path.resolve(__dirname, '../dist'),
      compress: true,
      inline: true, // 在构建变化后的代码会通过代理客户端来控制网页刷新
      host: '0.0.0.0',
      port: PORT,
      clientLogLevel: 'info',
      historyApiFallback: true,
      stats: 'errors-only',
      disableHostCheck: true,
    }
    // v5
    devServer: {
      // contentBase 变为 static 对象里面来配置
      static: {
        directory: path.resolve(__dirname, '../dist'),
      },
      client: {
        logging: 'info',
      },
      compress: true,
      // inline: true, // 直接移除,没有替代项
      host: '0.0.0.0',
      port: PORT,
      historyApiFallback: true,
      allowedHosts: 'all', // 代替 disableHostCheck: true
      // 新增中间件配置
      devMiddleware: {
        stats: 'errors-only',
      },
     },

相关文章:
webpack-dev-server v3 迁移 v4 指南

Webpack5 移除了 Node.js Polyfill(兼容)

Webpack5 移除了 Node.js Polyfill,将会导致一些包变得不可用(会在控制台输出 'XXX' is not defined),如果需要兼容 process/buffer 等 Nodejs Polyfill,则要安装相关的 Polyfill:process,并在 Plugin 中显式声明注入。业务代码中是有使用的process 变量的,故需要兼容,同时要安装process/buffer 库。

{
    plugins: [
        new webpack.ProvidePlugin({
          process: 'process/browser',
        }),
    ]
}

升级废弃的配置项

未改动前的项目配置项含义(config/webpack_prod.ts):

splitChunks: {
  chunks: 'all',
  minSize: 30000, // 模块要大于30kb才会进行提取
  minChunks: 1, // 最小引用次数,默认为1
  maxAsyncRequests: 5, // 异步加载代码时同时进行的最大请求数不得超过5个
  maxInitialRequests: 3, // 入口文件加载时最大同时请求数不得超过3个
  automaticNameDelimiter: '_', // 模块文件名称前缀
  name: true, // 自动生成文件名
  cacheGroups: {
    // 将来自node_modules的模块提取到一个公共文件中
    vendors: {
      test: /[\\/]node_modules[\\/]/,
      priority: -10, // 执行优先级,默认为0
    },
    // 其他不是node_modules中的模块,如果有被引用不少于2次,那么也提取出来
    default: {
      minChunks: 2,
      priority: -20,
      reuseExistingChunk: true, // 如果当前代码块包含的模块已经存在,那么不在生成重复的块
    },
  },
},
// v5
splitChunks: {
  cacheGroups: {
    // vendors ——> defaultVendors
    defaultVendors: {
      test: /[\\/]node_modules[\\/]/,
      priority: -10, // 执行优先级,默认为0
    },
  },
},

这个在(config/webpack_prod.ts)里面改个名字即可

splitChunks.name(移除)

splitChunks.name 表示抽取出来文件的名字

  • 在 v4 中该配置默认为 true 表示自动生成文件名
  • v5 移除了 optimization.splitChunks 的 name: true:不再支持自动命名

迁移:使用默认值。chunkIds: "named" 会为你的文件取一个有用的名字,以便于调试

output 配置(兼容)

(node:58337) [DEP_WEBPACK_TEMPLATE_PATH_PLUGIN_REPLACE_PATH_VARIABLES_HASH] DeprecationWarning: [hash] is now [fullhash] (also consider using [chunkhash] or [contenthash], see documentation for details)

hash已经不推荐使用了,改为了fullhash,这个名字比起原来的hash更清楚
config/webpack_dev.ts

// v4
output: {
    filename: 'js/[name].[hash].js',
    chunkFilename: 'js/[name].[hash].js',
},
// v5
output: {
    filename: 'js/[name].[fullhash].js',
    chunkFilename: 'js/[name].[fullhash].js',
},

优化配置项(废弃移除)

(config/webpack_prod.ts)

// webpack v4,在 v5 已被废弃,故需移除

new webpack.HashedModuleIdsPlugin()

HashedModuleIdsPlugin 作用:实现持久化缓存。模块 ID 通过 HashedModuleIdsPlugin来进行计算,它会把基于数字增量的 ID 替换成模块自身的 hash。这样的话,一个模块的 ID 只会在重命名或者移除的时候才会改变,新模块不会影响到它的 ID 变化。

webpack5 增加了确定的 moduleId,chunkId 的支持,如下配置:

optimization.moduleIds = 'deterministic'
optimization.chunkIds = 'deterministic'

此配置在生产模式下是默认开启的,它的作用是以确定的方式为 module 和 chunk 分配 3-5 位数字 id,替代 v4 版本的 HashedModuleIdsPlugin。

全局变量写法(兼容)

module.exports = () => {
  return {
    // ...
    plugins: [
      // webpack5 定义环境变量的写法变了
      new webpack.DefinePlugin({
        'process.env.NODE_ENV': JSON.stringify('production'),
      }),

    // webpack4的写法
    //   new webpack.DefinePlugin({
    //    'process.env': {
    //       NODE_ENV: JSON.stringify('production'),
    //     },
    //   }),
    ],
  };
};

使用cache属性,缓存进行优化(新增)

默认情况下 webpack5 不会启用持久化缓存,不自己手动加 cache 配置,webpack5等于没优化,直接升级二次构建反而更慢;并不是升级就自动提速,还是要手动配置
当设置 cache.type: "filesystem" 时,webpack 会在内部以分层方式启用文件系统缓存和内存缓存。 从缓存读取时,会先查看内存缓存,如果内存缓存未找到,则降级到文件系统缓存。 写入缓存将同时写入内存缓存和文件系统缓存。
文件系统缓存不会直接将对磁盘写入的请求进行序列化。它将等到编译过程完成且编译器处于空闲状态才会执行。 如此处理的原因是序列化和磁盘写入会占用资源,并且我们不想额外延迟编译过程。

cache: {
    // 将缓存类型设置为文件系统,默认为memory
    type: 'filesystem',

    buildDependencies: {
      // 当构建依赖的config文件(通过 require 依赖)内容发生变化时,缓存失效
      config: [__filename],
      // 如果你有其他的东西被构建依赖,你可以在这里添加它们
    },
},

为了防止缓存过于固定,导致更改构建配置无感知,依然使用旧的缓存,默认情况下,每次修改构建配置文件都会导致重新开始缓存。当然也可以自己主动设置 version 来控制缓存的更新。

最终优化成果

第一次编译:

第二次编译:

效果:二次构建提速85%以上

参考文章

03-05 23:47