webpack4 分包(splitChunks)

前言

用过webpack已经很久了,一直没有好好研究过分包,今天花点时间来好好研究下这个。

优势

打包优化: 公共代码提取,体积变小
用户体验优化:按需加载,下载更快,白屏更短

过程

module、chunk和bundle

要研究分包,首先要分清这三个概念

module: 模块,每个import 都是一个module模块

chunk: 依赖链路模块,我的个人理解,因为从一开始的entry配的入口文件开始,随着递归寻找其他的依赖文件,最后把所有的依赖串起来,这就是一个chunk。但是我们可以拆分多个chunk:

通过entry配置多入口文件
配置splitChunk支持同步、异步按需引入,使用同一个chunk

bundle: 就是指bundle,这个和chunk是一一对应的,只不过chunk是编译压缩前的,bundle是编译压缩后的。

分包方式

entry多入口打包

webpack配置

const { CleanWebpackPlugin } = require('clean-webpack-plugin')
const HtmlWebpackPlugin = require('html-webpack-plugin')

module.exports = {
  mode: 'none',
  entry: { // 将entry定义成一个对象,来设置多个打包入口
    index: './src/index.js',
    album: './src/album.js'
  },
  output: { // 修改输出文件名
    // 通过[name]这种占位符的方式动态输出文件名,[name]最终就会替换成打包入口名称
    filename: '[name].bundle.js'
  },
  module: {
    rules: [
      {
        test: /\.css$/,
        use: [
          'style-loader',
          'css-loader'
        ]
      }
    ]
  },
  plugins: [
    new CleanWebpackPlugin(),
    new HtmlWebpackPlugin({
      title: 'Multi Entry',
      template: './src/index.html',
      filename: 'index.html'
    }),
    new HtmlWebpackPlugin({
      title: 'Multi Entry',
      template: './src/album.html',
      filename: 'album.html'
    })
  ]
}

可以看到生成了多个 chunk\ bundle\ js文件
在这里插入图片描述

提取公共模块

在 index.js 和 alubm.js 中都有对公共模块的引入
在这里插入图片描述
可以将这些公共模块提取出来,在 webpack.config.js 中的 optimization 添加一个 splitChunks 进行配置

optimization: {
    splitChunks: {
       // 自动提取所有公共模块到单独 bundle
       chunks: 'all'
    }
  }

打包后会生成一个公共模块
在这里插入图片描述

splitChunks配置的chunks有三个值:async 、initial、all

async:chunks的默认值,提取异步加载的模块,打包成单独文件。
initial:提取异步和同步加载的模块(普通import不属于同步也不属于异步),打包成单独文件,就算这个文件被同步引入了,也被异步引入了,也会打包成两个文件。
all: 提取异步和同步加载的模块,就算这个文件被同步引入了,也被异步引入了,也会打包成一个文件。

举例:

async: loadsh这个包被多个文件异步引入,是会被打成一个公共包的
initial:  loadsh这个包被多个文件异步引入或被多个文件同步引入 ,是会被打成一个公共包的,如果同时被多个文件异步引入或和多个文件同步引入会被打成两个包
all:  loadsh这个包被多个文件异步引入或被多个文件同步引入 ,是会被打成一个公共包的,如果同时被多个文件异步引入或和多个文件同步引入也会被打成一个包
Webpack 中,SplitChunks 是一种非常强大的功能,用于将代码拆分成多个包(chunks),从而优化加载性能。通过合理配置 SplitChunks,可以减少初始加载时间、提高缓存效率,并改善用户体验。 SplitChunksPlugin 是 Webpack 内置的插件,无需额外安装即可使用。默认情况下,SplitChunks 只影响按需加载的 chunks(例如通过动态 `import()` 加载的模块)。可以通过 `optimization.splitChunks` 配置项来自定义分包策略。 ### SplitChunks 的核心配置项 - **chunks**: 指定哪些 chunks 应该被优化。可选值包括: - `'async'`:仅优化异步加载的 chunks(默认值)。 - `'initial'`:仅优化初始加载的 chunks。 - `'all'`:优化所有 chunks(包括同步和异步加载的)[^2]。 - **minSize**: 指定拆分 chunk 的最小大小(以字节为单位)。如果 chunk 的大小小于该值,则不会被拆分。 - **maxSize**: 指定拆分 chunk 的最大大小。如果 chunk 的大小超过该值,则会被进一步拆分。 - **minChunks**: 指定模块被引用的最小次数。只有被至少 `minChunks` 次引用的模块才会被拆分。 - **maxAsyncRequests**: 指定按需加载时的最大并行请求数。 - **maxInitialRequests**: 指定入口点的最大并行请求数。 - **automaticNameDelimiter**: 指定生成的 chunk 名称的分隔符。 - **name**: 指定生成的 chunk 的名称。可以是一个字符串或函数。 - **cacheGroups**: 定义自定义的缓存组,用于控制如何拆分特定类型的模块。例如,可以将所有第三方库提取到一个单独的 chunk 中。 ### 示例配置 以下是一个典型的 SplitChunks 配置,用于优化加载性能: ```javascript module.exports = { optimization: { splitChunks: { chunks: 'all', // 优化所有 chunks minSize: 20000, // 最小拆分大小 maxSize: 0, // 不限制最大大小 minChunks: 1, // 至少被引用一次就拆分 maxAsyncRequests: 30, // 异步加载的最大请求数 maxInitialRequests: 30, // 初始加载的最大请求数 automaticNameDelimiter: '~', // chunk 名称分隔符 name: true, // 自动生成 chunk 名称 cacheGroups: { // 缓存组:提取第三方库 vendors: { test: /[\\/]node_modules[\\/]/, priority: -10, chunks: 'all', }, // 缓存组:提取公共代码 default: { minChunks: 2, priority: -20, reuseExistingChunk: true, }, }, }, }, }; ``` ### 优化策略 1. **拆分第三方库**:通过 `cacheGroups` 中的 `vendors` 配置,可以将所有从 `node_modules` 加载的模块提取到一个单独的 chunk 中。这样可以利用浏览器缓存,减少重复下载。 2. **提取公共代码**:通过 `default` 缓存组,可以将多个 chunk 中重复使用的代码提取到一个公共 chunk 中,减少冗余代码。 3. **按需加载**:SplitChunks 可以优化按需加载的 chunks,确保异步加载的模块尽可能小,提高加载速度。 4. **控制 chunk 数量**:通过 `maxAsyncRequests` 和 `maxInitialRequests`,可以限制并行请求数量,避免过多的 HTTP 请求影响性能。 通过合理配置 SplitChunks,可以显著提升 Webpack 构建的性能,减少加载时间,并优化用户体验。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值