作用
虽然webpack4已经出来很久了,CommonsChunkPlugin也已经舍弃了,但是还是有很多项目是通过webpack3来打包的,对CommonsChunkPlugin各个配置项的认识还是比较模糊,这里做个详细的记录。

CommonsChunkPlugin通过将公共模块拆出来,最终合成的文件能够在最开始的时候加载一次,以便存到缓存中供后续使用。

    既然是将公共模块拆分出来,那么肯定是要有多个被打包的模块,即公共模块的来源,包括 入口文件和 已近提取出来的公共模块,如果来源只有一个,那么就立即被打包成一个单独文件,不会抽取公共模块,因为一个chunk来源是不会出现公共模块的,这是webpack打包策略,只会出现各个组件的相互引用,这种情况用于提取第三方库等;如果来源很多个,那么会提取出公共的模块组成一个单独的文件,来源文件再各自打包成各自的单独文件
    这里最重要的是搞清楚的就是配置项中各种chunk的概念和来源,我们首先看看有哪些配置选项。再对配置项逐一解析。

用法

new webpack.optimize.CommonsChunkPlugin(options);

options选项及官方翻译

{
  name: string, // or
  names: string[],
  /*name或者names是入口文件中的chunk名称,公共模块chunk可以在入口文件中定义名称和所对应的模块,然后被选择;
  *name和names的不同在于:name只选择一个入口chunk进行分开打包,names相当于name的遍历版,
  *对names数组中的chunk遍历,然后单独打包,此时options.filename不能是具体的名称,某则会出现相同的名称而报错,
  * */
  // 如果该选项被忽略,同时 `options.async` 或者 `options.children` 被设置,所有的 chunk 都会被使用,
  // 否则 `options.filename` 会用于作为 chunk 名。   filename: string,
  /*文件的名字模板,可以使用和output相同的占位符,如[hash:7],7代表7位hash符号,默认是20位;
  * 默认名字与`output.filename` 或者 `output.chunkFilename`相同*/   minChunks: number|Infinity|function(module, count) => boolean,
  // 在传入 公共chunk(commons chunk) 之前所需要包含的最少数量的 chunks 。
  // 数量必须大于等于2,或者少于等于 chunks的数量
  // 传入 `Infinity` 会马上生成 公共chunk,但里面没有模块。
  // 你可以传入一个 `function` ,以添加定制的逻辑(默认是 chunk 的数量)
  /*module.context代表chunk所以在的文件夹路劲,
  *module.resource代表该chunk的文件路劲
  *count chunk被引入的次数*/   chunks: string[],
  // 通过 chunk name 去选择 chunks 的来源。chunk 必须是 公共chunk 的子模块。
  // 如果被忽略,所有的,所有的 入口chunk (entry chunk) 都会被选择。
  /*chunks和children是相同的作用,不同的是chunks只选择特定的子模块抽取公共模块,
  *而children是选择所有的子模块*/   children: boolean,
  // 如果设置为 `true`,所有公共 chunk 的子模块都会被选择
  /*将子模块的公共模块打包进父 chunk */   deepChildren: boolean,
  // 如果设置为 `true`,所有公共 chunk 的后代(子孙)模块都会被选择   async: boolean|string,
  // 如果设置为 `true`,一个异步的 公共chunk 会作为 `options.name` 的子模块,和 `options.chunks` 的兄弟模块被创建。
  // 它会与 `options.chunks` 并行被加载。   minSize: number,
  // 在 公共chunk 被创建立之前,所有 公共模块 (common module) 的最少大小。
}

配置项介绍
不按照上面的顺序介绍,有些配置项很相似,容易弄混,按照我自己理解来解释。

1. filename
这是文件名的模板,和output.filename 或者 output.chunkFilename是一个意思,最终生成的文件名。

如果配置了这个项,以这个模板优先取名,否则,按照output.filename 或者 output.chunkFilename规则取名。

2. name or names
这个抽取的公共模块的名字,这个和filename还是有区别的,filename=name+hash+其它可能的配置项+ext。
example1: 这里adminA和adminB都引入的common.js,所以会抽取common.js作为公共模块。

//adminA
require("./common")
console.log("adminA")
//adminA
require("./common")
console.log("adminB")
entry: {
  adminA: "./adminA",
  adminB: "./adminB",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
},
plugins: [
  new webpack.optimize.CommonsChunkPlugin({
    name: "commons",
  }),
]

打包生成名为 commons的公共模块

同时,如果这个name和入口文件的name相同,那么入口文件直接被选择为打包来源,即直接指定name所指定的入口文件为公共模块;这个在很明确公共模块是什么的时候很有用,比如第三方库,可以直接指定。
names则是通过遍历names数组,执行多次该插件,免去写多个相同的代码(如果有这个需求)。例:
example2: 增加adminC,但是没有引入common.js,只有adminA和adminB引入了。

//adminC
console.log("adminC");
entry: {
  adminA: "./adminA",
  adminB: "./adminB",
  adminC: "./adminC",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
},
plugins: [
  new BundleAnalyzerPlugin(),
  new webpack.optimize.CommonsChunkPlugin({
    name: "commons",
  }),
]

虽然也生成了commons.js但是其实里面没有任何东西(文件很小),adminA和adminB文件还是很大,也就是没有抽取公共模块:

可以用插件webpack-bundle-analyzer分析:common.js还是存在adminA.js和adminB.js里面

这个时候在入口文件直接指定公共模块commons:
example3

entry: {
adminA: "./adminA",
adminB: "./adminB",
adminC: "./adminC",
//name和配置项的name相同
commons: './common.js'
},

可以从文件大小看出来,公共模块已经抽取出来了。

3. minChunks
被其他模块引用的次数 (默认是3),只有满足这个引入次数条件才会被抽取,如果设置成 Infinity (无限大,不可能满足),那么会立即打包成一个独立模块,不会抽取公共模块,用于公共模块很明确的时候或者想直接打包某一chunk。

例如前面的example2,也可以不用在入口指定公共模块,主要是因为minChunk的默认值是3,而common.js只被adminA和adminB引入了,也就是只引入了两次,所以不会被打包,我们这里将example2的minChunks设置2,再打包:
example4:

entry: {
  adminA: "./adminA",
  adminB: "./adminB",
  adminC: "./adminC",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
},
plugins: [
  new BundleAnalyzerPlugin(),
  new webpack.optimize.CommonsChunkPlugin({
    name: "commons",
    minChunks: 2
  }),
]

从文件大小上可以看出,common.js已经抽取出来了:

minChunks也可以是一个函数,webpack会遍历所有来源chunk,逐一执行这个函数,函数返回true这个chunk才被提取为公共模块。这个函数用于更加精确的提取公共模块,如vue构建工具生成的配置文件,可以提取所有在文件夹node_modules中的模块:
example5:

//这里是提取所有的第三方插件,如我这里有vue,vue-router,echarts
//但是echarts太庞大了,需要和vue,vuer-router分离
new webpack.optimize.CommonsChunkPlugin({
  name: 'vendor',
  minChunks(module) {
    /*module.context代表chunk所以在的文件夹路劲,
    *module.resource代表该chunk的文件路劲
    *count chunk被引入的次数*/
    return (
      module.resource &&
      /\.js$/.test(module.resource) &&
      module.resource.indexOf(
        path.join(__dirname, '../node_modules')
      ) === 0
    )
  },
  chunks: ["app"]
}),
//分离vue,vue-router,echarts是按需引入的,不太方便按这种方式抽取。
new webpack.optimize.CommonsChunkPlugin({
  name: 'vue',
  minChunks: function (module) {
    return (module.resource && /\.js$/.test(module.resource) && (/vue/).test(module.resource))
  },
  chunks: ['vendor']
}),

4. chunks

选择chunk来源,即选择要被抽取的模块来源,这个很好理解,即被选择的chunk来源才会被抽取公共模块。如:A和B同时引入C,而D没有引入C,那么就可以只选择A和B作为chunk来源,抽取A、B的公共模块C,否则C可能不会被抽取为公共模块。
eample6: adminA、adminB和adminC都引入common.js,但是chunks只选择adminA和B。

entry: {
  adminA: "./adminA",
  adminB: "./adminB",
  adminC: "./adminC",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
},
plugins: [
  new BundleAnalyzerPlugin(),
  new webpack.optimize.CommonsChunkPlugin({
    name: "commons",
    minChunks: 2,
    chunks: ['adminA', 'adminB']
  }),
]

结果:adminA和adminB的common.js被抽取出来了,但是adminC的common.js没有被抽取出来(从大小上判断)。

5. children
children: webpack官方给出了一个案例,中文版是翻译过来的,但是我看了很久都没看明白,其实翻译是不够严谨的,案例中翻译说 一个 chunk的子模块会有公共模块,配置参数里面翻译是公共模块的子模块会有公共依赖,我想了好久,都没想出这个子模块是什么意思;其实是 一个入口的模块的子模块,子模块指的是 异步加载模块、通过代码分割功能从这个入口chunk分离出来的chunk,在模块中使用import()或者require.ensure(),会单独打包这一部分模块。

eample7: adminA通过代码分割功能,异步加载childA和childB,childA和childB都引入了jquery和common。

//adminA.js
import('./childA');
import('./childB');
console.log("adminA");
//chuildA
import './jquery';
import './common';
console.log('childA');
//chuildB
import './jquery';
import './common';
console.log('childB');
//webpack.config.js

entry: {
  adminA: "./adminA",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js",
  chunkFilename: '[name].chunk.js'
},
plugins: []

不优化的打包结果:生成了两个异步的chunk,0和1,就是childA和childB。childA和childB都引入了common.js和jquery.js,体积很大。

我们试着优化,看能不能把异步加载的模块的公共模块抽取出来:
eample8:

//webpack.config.js
entry: {
  adminA: "./adminA",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js",
  chunkFilename: '[name].chunk.js'
},
plugins: [
  new webpack.optimize.CommonsChunkPlugin({
    name: "commons",
    minChunks: 2
  }),
]

结果:common.js和jquery.js并没有抽取出来成为公共模块,只是抽取了webpack代码分离功能的代码,因为代码分离出来的chunk,并不在chunk来源选择范围内(忽略了chunks选项,那么默认是全部的chunk来源,但是排除代码分离的chunk)。

我们加上children:true,并且需要把name设置成入口chunk的name,此处是adminA:

eample9:

//webpack.config.js
entry: {
  adminA: "./adminA",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js",
  chunkFilename: '[name].chunk.js'
},
plugins: [
  new webpack.optimize.CommonsChunkPlugin({
    name: "adminA",
    minChunks: 2,
    children: true
  }),
]

结果:common.js和jquery.js被打包进了adminA,也就是代码分离后子模块的公共模块被打包进了父模块。减少了总体体积。但是会增加初始加载时间(即加载adminA的时间会延长),如果分离子模块的公共代码很多,并且预计到用户会加载很多子模块(代码分离的子模块),可以这样做。当然,还有另外一种方法,就是async.

6.async

async: 就是把代码分离模块的公共模块抽取出来。不同于上面children:true是把公共模块抽取放到父模块中,async:true把公共模块单独抽取出来作为一个公共模块,和之前抽取公共模块是一样的,只不过这个抽取的是分离代码的公共模块。

再看**example10:**在example9的基础上加上async,async可以取值Boolean值也可以是string,如果是string,那么这个string值作为chunk打包后的name,如果是Boolean值,那么打包后的name就是0。

同样,这里的options.name要和入口chunk的name匹配

entry: {
  adminA: "./adminA",
},
output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js",
  chunkFilename: '[name].chunk.js'
},
plugins: [
  new BundleAnalyzerPlugin(),
  new webpack.optimize.CommonsChunkPlugin({
    //要和入口chunk的name匹配
    name: "adminA",
    minChunks: 2,
    children: true,
    async: "async-chunk"
  }),
]

名为async-chunk.chunk.js就是我们分离代码的公共模块。

7.minSize
抽取公共代码之前,应该满足的文件的大小。如果抽取出来的公共代码文件大小只有几kb,这样会增加请求次数,还不如不抽取。

8.deepChildren
这个目前还不知道有什么用,和children差不多,children选择的是直接子代,deepChildren选择的是所有后代,但是目前我并没有发现它有什么用。

原文链接:https://blog.csdn.net/zy444263/article/details/85252477

[转]webpack——CommonsChunkPlugin的更多相关文章

  1. webpack CommonsChunkPlugin详细教程

    1.demo结构: 2.package.json配置: { "name": "webpack-simple-demo", "version" ...

  2. [Webpack 2] Chunking common modules from multiple apps with the Webpack CommonsChunkPlugin

    If you have a multi-page application (as opposed to a single page app), you’re likely sharing module ...

  3. [Webpack 2] Grouping vendor files with the Webpack CommonsChunkPlugin

    Often, you have dependencies which you rarely change. In these cases, you can leverage the CommonsCh ...

  4. webpack CommonsChunkPlugin 提取公共代码

    1.项目结构 2.部分代码 module.js console.log('module.js'); index文件夹下的index.js require('../module.js'); consol ...

  5. Webpack+React+ES6开发模式入门指南

    React无疑是今年最火的前端框架,github上的star直逼30,000,基于React的React Native的star也直逼20,000.有了React,组件化似乎不再步履蹒跚,有了Reac ...

  6. Webpack+React+ES6入门指南[转]

    React无疑是今年最火的前端框架,github上的star直逼30,000,基于React的React Native的star也直逼20,000.有了React,组件化似乎不再步履蹒跚,有了Reac ...

  7. webpack 多页应用架构系列实战

    阅读目录 1.webpack配置了解 2.webpack CommonsChunkPlugin公共代码剥离 3.了解ProvidePlugin的用途 回到顶部 1.webpack配置了解 webpac ...

  8. React的React Native

    React的React Native React无疑是今年最火的前端框架,github上的star直逼30,000,基于React的React Native的star也直逼20,000.有了React ...

  9. 关于webpack.optimize.CommonsChunkPlugin的使用二

    Note:当有多个入口节点的时候,只有所有入口节点都引入了同一个模块的时候,webpack.optimize.CommonsChunkPlugin才会将那个模块提取出来,如果其中一个入口节点没有引入该 ...

随机推荐

  1. java多线程5:线程间的通信

    在多线程系统中,彼此之间的通信协作非常重要,下面来聊聊线程间通信的几种方式. wait/notify 想像一个场景,A.B两个线程操作一个共享List对象,A对List进行add操作,B线程等待Lis ...

  2. Android App加固原理与技术历程

    App为什么会被破解入侵 随着黑客技术的普及化平民化,App,这个承载我们移动数字工作和生活的重要工具,不仅是黑客眼中的肥肉,也获得更多网友的关注.百度一下"App破解"就有529 ...

  3. 记一次AWD

    有幸bjx师傅又让我参加了一次awd,算是第二次体验awd,又感觉学习到了很多东西. 第一次打这种模式的时候,我几乎什么都没有做,就给师傅们下载文件,上传文件了.(太菜了) 昨晚分的组,发现没有人是p ...

  4. 联盛德 HLK-W806 (九): 软件SPI和硬件SPI驱动ST7789V液晶LCD

    目录 联盛德 HLK-W806 (一): Ubuntu20.04下的开发环境配置, 编译和烧录说明 联盛德 HLK-W806 (二): Win10下的开发环境配置, 编译和烧录说明 联盛德 HLK-W ...

  5. Python pyecharts绘制水球图

    一.水球图Liquid.add()方法简介 Liquid.add()方法签名add(name, data, shape='circle', liquid_color=None, is_liquid_a ...

  6. 成本资源(Project)

    <Project2016 企业项目管理实践>张会斌 董方好 编著 张同学说,成本资源就是balabalabala--算了,反正就是一种资源,比如,张同学列出的差旅费.住宿费.交通费.通信费 ...

  7. linux安装软件系列之yum安装

    自动搜索最快镜像插件: yum install yum-fastestmirror 安装yum图形窗口插件: yum install yumex 1.安装 yum install 全部安装 yum i ...

  8. .NET Core基础篇之:白话管道中间件

    在.Net Core中,管道往往伴随着请求一起出现.客户端发起Http请求,服务端去响应这个请求,之间的过程都在管道内进行. 举一个生活中比较常见的例子:旅游景区. 我们都知道,有些景区大门离景区很远 ...

  9. 如何获取网管MTU

    在本机打开dos窗口,执行: ping -f -l 1472 192.168.0.1 其中192.168.0.1是网关IP地址,1472是数据包的长度.请注意,上面的参数是"-l" ...

  10. npm 包发布,自己本机发布,前端内部发布,全网发布

    第一步,安装 sinopia npm install -g sinopia 开启终端一: 第二步,启动 sinopia -l 127.0.0.1:4873 开启终端二: cd 到某个指定仓库 mkdi ...