高级配置
# 高级配置
基本配置主要确保项目能够在 demo 环境运行,而用于线上环境时往往需要进行一些高级配置。
Webpack 的高级配置主要分为 6 个方面,根据需要进行选择:
- 多入口
- 抽离压缩 CSS 文件
- 抽离公共代码
- 懒加载 - 异步加载 JS
- 处理 JSX
- 处理 Vue
# 多入口
通过基本配置,我们在打包编译后产生的页面只是一个文件 index.html
。如果在一个项目中想产生两个页面 index.html
和 other.html
(或多个页面),就需要进行多入口配置。
首先在 webpack.common.js
中建入口(entry
)的时候就需要建立两个(或多个),其次在插件(plugins
)中针对每一个入口都要 new
一个 HtmlWebpackPlugin
的实例。配置代码如下:
const path = require('path')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const { srcPath, distPath } = require('./paths')
module.exports = {
entry: {
index: path.join(srcPath, 'index.js'),
other: path.join(srcPath, 'other.js')
},
module: {
rules: [
// 此处省略处理 js、css、less 的配置
]
},
plugins: [
// 多入口 - 生成 index.html
new HtmlWebpackPlugin({
template: path.join(srcPath, 'index.html'), // 对应到 index.html 这个模板文件
filename: 'index.html', // 生成的文件,名字随便取
// chunks 表示该页面要引入哪些 JS 文件(即 entry 中配置的 JS 文件),默认全部引用
chunks: ['index'] // 只引用 index.js
}),
// 多入口 - 生成 other.html
new HtmlWebpackPlugin({
template: path.join(srcPath, 'other.html'), // 对应到 other.html 这个模板文件
filename: 'other.html', // 生成的文件,名字随便取
chunks: ['other'] // 只引用 other.js
})
]
}
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
30
其次修改 prod(生产环境配置)中的 output
,将 webpack.prod.js
中 output.filename
的固定值 bundle
修改为变量 [name]
。配置代码如下:
const path = require('path')
const webpack = require('webpack')
const { CleanWebpackPlugin } = require('clean-webpack-plugin')
const webpackCommonConf = require('./webpack.common.js')
const { merge } = require('webpack-merge')
const { srcPath, distPath } = require('./paths')
module.exports = merge(webpackCommonConf, {
mode: 'production',
output: {
// filename: 'bundle.[contenthash:8].js', // 打包代码时,加上 hash 戳
filename: '[name].[contenthash:8].js', // name 即多入口时 entry 的 key
path: distPath,
},
module: {
rules: [
// 此处省略处理图片的配置
]
},
plugins: [
new CleanWebpackPlugin(), // 会默认清空 output.path 文件夹
new webpack.DefinePlugin({
// window.ENV = 'production'
ENV: JSON.stringify('production')
})
]
})
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
# 抽离压缩 CSS 文件
在基础配置中,是把所有 CSS 文件全部写到页面的 style 标签里,这种方式在开发模式中问题不大,但在生产环境中显然不科学,因此我们需要把 CSS 文件抽离压缩。
首先在 webpack.common.js
中删除对 CSS 和 Less 的处理,只保留对 JS 文件的处理:
const path = require('path')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const { srcPath, distPath } = require('./paths')
module.exports = {
entry: {
index: path.join(srcPath, 'index.js')
},
module: {
rules: [
{
test: /\.js$/,
use: ['babel-loader'],
include: srcPath,
exclude: /node_modules/
}
]
},
plugins: [
new HtmlWebpackPlugin({
template: path.join(srcPath, 'index.html'),
filename: 'index.html'
})
]
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
其次将原来 CSS 和 Less 的处理逻辑放到开发环境下,即 webpack.dev.js
中:
const path = require('path')
const webpack = require('webpack')
const webpackCommonConf = require('./webpack.common.js')
const { merge } = require('webpack-merge')
const { srcPath, distPath } = require('./paths')
module.exports = merge(webpackCommonConf, {
mode: 'development',
module: {
rules: [
// 直接引入图片 url
{
test: /\.(png|jpg|jpeg|gif)$/,
use: 'file-loader'
},
{
test: /\.css$/,
// loader 的执行顺序是:从后往前
use: ['style-loader', 'css-loader', 'postcss-loader'] // 加了 postcss
},
{
test: /\.less$/,
// 增加 'less-loader' ,注意顺序
use: ['style-loader', 'css-loader', 'less-loader']
}
]
},
plugins: [
new webpack.DefinePlugin({
// window.ENV = 'development'
ENV: JSON.stringify('development')
})
],
devServer: {
// 此处省略 devServer 的配置
}
})
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
30
31
32
33
34
35
36
37
然后才是在 webpack.prod.js
中进行配置,这里对 CSS 和 Less 的处理逻辑与在开发环境中很不一样。需要安装三个插件 —— 用于抽离的 mini-css-extract-plugin
,用于压缩的 terser-webpack-plugin
和 optimize-css-assets-webpack-plugin
。配置代码如下:
const path = require('path')
const webpack = require('webpack')
const { merge } = require('webpack-merge')
const { CleanWebpackPlugin } = require('clean-webpack-plugin')
const MiniCssExtractPlugin = require('mini-css-extract-plugin')
const TerserJSPlugin = require('terser-webpack-plugin')
const OptimizeCSSAssetsPlugin = require('optimize-css-assets-webpack-plugin')
const webpackCommonConf = require('./webpack.common.js')
const { srcPath, distPath } = require('./paths')
module.exports = merge(webpackCommonConf, {
mode: 'production',
output: {
filename: '[name].[contenthash:8].js', // name 即多入口时 entry 的 key
path: distPath,
},
module: {
rules: [
// 图片 - 考虑 base64 编码的情况
{
test: /\.(png|jpg|jpeg|gif)$/,
use: {
loader: 'url-loader',
options: {
// 小于 5kb 的图片用 base64 格式产出
// 否则,依然延用 file-loader 的形式,产出 url 格式
limit: 5 * 1024,
// 打包到 img 目录下
outputPath: '/img/',
// 设置图片的 cdn 地址(也可以统一在外面的 output 中设置,那将作用于所有静态资源)
// publicPath: 'http://cdn.abc.com'
}
}
},
// 抽离 css
{
test: /\.css$/,
use: [
MiniCssExtractPlugin.loader, // 注意,这里不再用 style-loader
'css-loader',
'postcss-loader'
]
},
// 抽离 less
{
test: /\.less$/,
use: [
MiniCssExtractPlugin.loader, // 注意,这里不再用 style-loader
'css-loader',
'less-loader',
'postcss-loader'
]
}
]
},
plugins: [
new CleanWebpackPlugin(), // 会默认清空 output.path 文件夹
new webpack.DefinePlugin({
// window.ENV = 'production'
ENV: JSON.stringify('production')
}),
// 抽离 css 文件,命名为 main + hash值
new MiniCssExtractPlugin({
filename: 'css/main.[contenthash:8].css'
})
],
optimization: {
// 压缩 css
minimizer: [new TerserJSPlugin({}), new OptimizeCSSAssetsPlugin({})],
}
})
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
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
# 抽离公共代码
为了提升性能,我们需要在打包时将第三方模块和公用引用的代码单独拆分出去。
既然是在打包时进行的优化,就是修改首先在 webpack.prod.js
这份生产环境的配置文件。在 optimization
中添加分割代码块的逻辑,配置代码如下:
const webpack = require('webpack')
const { merge } = require('webpack-merge')
const { CleanWebpackPlugin } = require('clean-webpack-plugin')
const MiniCssExtractPlugin = require('mini-css-extract-plugin')
const TerserJSPlugin = require('terser-webpack-plugin')
const OptimizeCSSAssetsPlugin = require('optimize-css-assets-webpack-plugin')
const webpackCommonConf = require('./webpack.common.js')
const { srcPath, distPath } = require('./paths')
module.exports = merge(webpackCommonConf, {
mode: 'production',
output: {
filename: '[name].[contenthash:8].js', // name 即多入口时 entry 的 key
path: distPath,
},
module: {
rules: [
// 图片 - 考虑 base64 编码的情况
{
test: /\.(png|jpg|jpeg|gif)$/,
use: {
loader: 'url-loader',
options: {
// 小于 5kb 的图片用 base64 格式产出
// 否则,依然延用 file-loader 的形式,产出 url 格式
limit: 5 * 1024,
// 打包到 img 目录下
outputPath: '/img/',
// 设置图片的 cdn 地址(也可以统一在外面的 output 中设置,那将作用于所有静态资源)
// publicPath: 'http://cdn.abc.com'
}
}
},
// 抽离 css
{
test: /\.css$/,
use: [
MiniCssExtractPlugin.loader, // 注意,这里不再用 style-loader
'css-loader',
'postcss-loader'
]
},
// 抽离 less
{
test: /\.less$/,
use: [
MiniCssExtractPlugin.loader, // 注意,这里不再用 style-loader
'css-loader',
'less-loader',
'postcss-loader'
]
}
]
},
plugins: [
new CleanWebpackPlugin(), // 会默认清空 output.path 文件夹
new webpack.DefinePlugin({
// window.ENV = 'production'
ENV: JSON.stringify('production')
}),
// 抽离 css 文件
new MiniCssExtractPlugin({
filename: 'css/main.[contenthash:8].css'
})
],
optimization: {
// 压缩 css
minimizer: [new TerserJSPlugin({}), new OptimizeCSSAssetsPlugin({})],
// 分割代码块
splitChunks: {
chunks: 'all',
/**
* initial 入口 chunk,对于异步导入的文件不处理
* async 异步 chunk,只对异步导入的文件处理
* all 全部 chunk,一般选择 all 模式
*/
// 缓存分组
cacheGroups: {
// 第三方模块
vendor: {
name: 'vendor', // chunk 名称
priority: 1, // 权限更高,优先抽离(例如第三方模块同时也作为公共模块在多处引用时,按第三方模块的规则进行抽离)
test: /node_modules/, // 检查模块是否位于 node_modules/ 目录下
minSize: 30000, // 大小限制(Byte),太小的不用抽离
minChunks: 1 // 最少复用过几次(第三方模块只要引用过一次就抽取出来)
},
// 公共的模块
common: {
name: 'common', // chunk 名称
priority: 0, // 优先级
minSize: 50000, // 公共模块的大小限制(此处设置 50KB)
minChunks: 2 // 公共模块最少复用过几次
}
}
}
}
})
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
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
对于多入口的情况,为了防止多余的第三方模块被打包到没有引用它的页面里,需要在 webpack.common.js
中的 plugins
内选择所需的 chunk,代码如下:
const path = require('path')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const { srcPath, distPath } = require('./paths')
module.exports = {
entry: {
index: path.join(srcPath, 'index.js'),
other: path.join(srcPath, 'other.js')
},
module: {
rules: [
{
test: /\.js$/,
use: ['babel-loader'],
include: srcPath,
exclude: /node_modules/
}
]
},
plugins: [
// 多入口 - 生成 index.html
new HtmlWebpackPlugin({
template: path.join(srcPath, 'index.html'),
filename: 'index.html',
// chunks 表示该页面要引用哪些 chunk (即上面的 index 和 other),默认全部引用
chunks: ['index', 'vendor', 'common'] // 要考虑代码分割
}),
// 多入口 - 生成 other.html
new HtmlWebpackPlugin({
template: path.join(srcPath, 'other.html'),
filename: 'other.html',
chunks: ['other', 'common'] // 考虑代码分割
})
]
}
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
30
31
32
33
34
35
目前 chunks 一共出现在了三个地方:
- common 里的
entry
定义了要生成哪些 chunk - common 里的
plugins
中定义了某个页面要引用哪些 chunk - prod 里的
splitChunks
中定义了代码分割成哪些 chunk
# 懒加载 - 异步加载 JS
懒加载就是引入动态数据,webpack 本身支持这种机制,所以不需要额外配置,只需要借助 import()
语法来引入 JS 文件。这种语法和 Vue 和 React 中的异步组件是一样的。
如下例子,这段代码写在某个 JS 文件中,将在 1.5s 之后加载出 dynamic-data.js
这个文件:
setTimeout(() => {
// 定义 chunk
import('./dynamic-data.js').then(res => {
console.log(res.default.message)
})
}, 1500)
2
3
4
5
6
这种异步加载也会产出一个 chunk,文件名是随机的字符串。
# 处理 JSX
借助 babel 即可,参考官网 (opens new window)。
需要先安装 @babel/preset-react
:
npm install --save-dev @babel/preset-react
然后在 .babelrc
中写上 @babel/preset-react
:
{
"presets": ["@babel/preset-react"]
}
2
3
# 处理 Vue
借助 vue-loader 即可,参考官网 (opens new window)。
需要先安装依赖包:
npm i vue-loader
然后在 webpack.common.js
中添加对应的规则:
const path = require('path')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const { srcPath, distPath } = require('./paths')
module.exports = {
entry: path.join(srcPath, 'index'),
module: {
rules: [
{
test: /\.js$/,
use: ['babel-loader'],
include: srcPath,
exclude: /node_modules/
},
{
test: /\.vue/,
use: ['vue-loader'],
include: srcPath
}
]
},
plugins: [
new HtmlWebpackPlugin({
template: path.join(srcPath, 'index.html'),
filename: 'index.html'
})
]
}
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
(完)