webpack中的hash、chunkhash、contenthash分别是什么

知人者智,自知者明,胜人者有力,自胜者强。——老子

简介

webpack中有三种hash可以配置,分别是hashchunkhashcontenthash他们是不对的可以针对不同的配置,首相要搞清楚这三种的hash的区别,什么场景下,适合用哪种。

hash
所有文件哈希值相同,只要改变内容跟之前的不一致,所有哈希值都改变,没有做到缓存意义

chunkhash
当有多个chunk,形成多个bundle时,如果只有一个chunk和一个bundle内容变了,其他的bundlehash都会发生变化,因为大家都是公用的一个hash,这个时候chunkhash的作用就出来了。它根据不同的入口文件(Entry)进行依赖文件解析、构建对应的 chunk,生成对应的哈希值

contenthash
在打包的时候我们会在js中导入css文件,因为他们是同一个入口文件,如果我只改了js得代码,但是他的css抽取生成css文件时候hash也会跟着变换。这个时候contenthash的作用就出来了。

下面直接用代码验证上面的猜想。

一个简单的 webpack 配置

我们现在就只创建一个能编译jswebpack配置,步骤如下:

  1. 创建一个空文件加,并且在当文件夹中打开 bash or cmd
  2. npm init -y 生成package.json
  3. 如果你安装了cnpm or yarn 就执行 cnpm i webpack webpack-cli -D, 安装webpack的包。
  4. 创建src,在src内部创建chunk0.jschunk1.jscommon.jsindex.jsstyle.css,并且编写内部代码
  5. 在项目根目录创建 webpack.config.js
  6. 直接在cmd中运行 webpack

文件目录如下:
webpack contenthash hash chunkhash

下面是代码
chunk0.js

1
2
3
export default function add(a, b) {
return a + b;
}

chunk1.js

1
2
3
export default function flow() {
return 'flow';
}

common.js

1
2
3
export default function commonJs() {
return 'commonJs';
}

index.js

1
2
3
4
import add from './chunk0.js';
import commonJs from './common';
console.log(add(1, 2));
console.log(commonJs());

style.css

1
2
3
body {
background: #000;
}

webpack.config.js

1
2
3
4
5
6
7
8
9
10
module.exports = {
mode: 'production', // 如果不添加就会警告
entry: {
index: './src/index.js', // 一个入口文件
chunk1: './src/chunk1.js' // 两一个入口文件
},
output: {
filename: '[name].[hash].js' // 出口文件
}
};

hash

我们直接运行webpack,运行结果如下图所示:

只有一个 hash,所有文件的 hash 都是相同:

webpack contenthash hash chunkhash
如果我们改变修改chunk1.js中的代码:

1
2
3
export default function flow() {
return 'flow1'; // flow => folw1
}

再运行 webpack 发现所有的 hash 都变化了,如下图所示:

webpack contenthash hash chunkhash
webpack contenthash hash chunkhash

对比发现他们的 hash 并不相同了,这个时候如果想修改了chunk1.js,index.js 不产生变化,就要用到 chunkhash。

chunkhash

  • 第一步 我们先把webpack.config.js做一下修改
1
2
3
4
5
6
7
8
9
10
module.exports = {
mode: 'production',
entry: {
index: './src/index.js',
chunk1: './src/chunk1.js'
},
output: {
filename: '[name].[chunkhash].js' // hash => chunkhash
}
};
  • 第二步 我们运行webpack

webpack contenthash hash chunkhash
根据上面图片发下,两个chunkhash并不相同了。

  • 第三部 我们修改 chunk1.js
1
2
3
export default function flow() {
return 'flow11111'; // flow1 => flow11111
}
  • 再运行webpack

webpack contenthash hash chunkhash
webpack contenthash hash chunkhash

根据图片我们看到了chunk1.jshash变化,而index.jshash并没有变化,达到了我们预期的效果,对我们线上的缓存也是比较好的。

contenthash

但是当我们一个js文件里面引用了一个css文件,如果我么修改了css文件内的内容,我们css中的内容,会发发现这整个bundlehash也会发生更新。
我们要引入css,并且要把css提出、压缩生成一个css文件,就要借助一个webpack的插件,叫做MiniCssExtractPlugin,他可以帮我提取 css 到 css 文件,并且压缩 css。

  • 第一步先安装css-loadermini-css-extract-plugin
1
cnpm install css-loader mini-css-extract-plugin -D
  • 第二步修改webpack.config.js 如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
const MiniCssExtractPlugin = require('mini-css-extract-plugin'); // 新增
module.exports = {
mode: 'production',
entry: {
index: './src/index.js',
chunk1: './src/chunk1.js'
},
output: {
filename: '[name].[chunkhash].js'
},
module: {
// 新增
rules: [
{
test: /\.css$/,
use: [MiniCssExtractPlugin.loader, 'css-loader']
}
]
},
plugins: [
// 新增
// 提取css插件
new MiniCssExtractPlugin({
// Options similar to the same options in webpackOptions.output
// both options are optional
filename: '[name].[chunkhash].css'
})
]
};
  • 第三步运行 webpack

webpack contenthash hash chunkhash

看代码可以看到index.cssindex.jshash是一样的。

  • 第四步修改 style.css
1
2
3
html {
font-size: 13px;
}
  • 第五步运行 webpack

webpack contenthash hash chunkhash
webpack contenthash hash chunkhash

对比两次构建的hash,发现只修改了style.css的文件,引入他的index.js确也更新了hash,这个时候就需要contenthash来发挥作用了。

  • 第六步修改webpack.config.js 并且运行 webpack
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
const MiniCssExtractPlugin = require('mini-css-extract-plugin'); // 新增
module.exports = {
mode: 'production',
entry: {
index: './src/index.js',
chunk1: './src/chunk1.js'
},
output: {
filename: '[name].[chunkhash].js'
},
module: {
// 新增
rules: [
{
test: /\.css$/,
use: [MiniCssExtractPlugin.loader, 'css-loader']
}
]
},
plugins: [
// 新增
// 提取css插件
new MiniCssExtractPlugin({
// Options similar to the same options in webpackOptions.output
// both options are optional
filename: '[name].[contenthash].css'
})
]
};

webpack contenthash hash chunkhash

看到他们直接 hash 就是不同的。

  • 修改common.js,直接运行 webpack
1
2
3
export default function commonJs() {
return 'commonJs1';
}

webpack contenthash hash chunkhash
webpack contenthash hash chunkhash

看到修改 js 时我们的 css 文件的 hash 并没有变更。

注意,当使用contenthash时,如果修改 js 文件,css 文件的 hash 不会变化,但是修改 js 的文件,css 文件的 hash 也会变化。

总结

hash 所有文件哈希值相同;
chunkhash 根据不同的入口文件(Entry)进行依赖文件解析、构建对应的 chunk,生成对应的哈希值;
contenthash 计算与文件内容本身相关,主要用在 css 抽取 css 文件时。

参考

面试必备!webpack 中那些最易混淆的 5 个知识点