由于新建项目发版打包时间大概需要「30分钟」,发版时「严重」拖慢「下班」时间,所以特意查看了相关文档来优化打包速度,争取早点下班,。 要优化,先分析。我们先要知道到底是哪里拖慢我们的打包速度呢? 可以利用webpack-bundle-analyzer插件来分析我们打包后生成的文件 修改webpack.prod.conf.js文件 通过图片可以看到打包后文件的具体信息 simple-progress-webpack-plugin可以显示打包百分比 修改webpack.prod.conf.js文件 效果如下: 通过上面进度可以看到,打包过程中,卡顿在压缩的地方过长,当项目越来越臃肿的时候,我们要需要对项目静态资源以及依赖包进行整理, 项目里面使用ElementUI和Echarts都是全部引用挂在Vue.prototype上,现都改为按需引用。 resolve.extensions能够自动解析确定的扩展,但是如果extensions扩展名过多,会导致解析过程过多,所以我们要合理配置扩展名,不要过多配置扩展名,项目引用多的文件,扩展名放在前面,我司项目中多的是vue,js文件,可以只引用这两种。 预处理各种文件时指定匹配目录后,webpack解析文件时就不会循环查找其他目录,加快解析速度。 webpack执行预处理文件时单线程的,我们可以使用happypack来多线程处理文件。 修改webpack.base.js文件 babel-plugin-dynamic-import-node插件是使import() 替换成 require 编译 修改.babelrc 文件 「注意」:使用插件build后没有chunk files文件。 通过DllPlugin插件分离出第三方包 使用add-asset-html-webpack-plugin动态添加dll.js到html。 「需要注意」 使用 externals 配置来告诉 webpack 不用打包第三方库,需要在 index.html 页面引入第三方库的 CDN 项目经过以上优化,打包从「30」分钟,到「2」分钟不到,整体还有优化空间,可以使用其他cdn等优化方式。打包优化方案
分析打包文件
打包后生成文件分析
npm i webpack-bundle-analyzer -D
const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin
// 构建完成后,浏览器会自动打开localhost:8080
webpackConfig.plugins.push(
new BundleAnalyzerPlugin({
analyzerPort: 8080,
generateStatsFile: false
})
)
打包进度条显示,可以查看到打包进度百分比
npm i simple-progress-webpack-plugin -D
const SimpleProgressWebpackPlugin = require( 'simple-progress-webpack-plugin' )
...
plugins: [
new SimpleProgressWebpackPlugin()
]
...
资源与依赖包的控制
减少文件搜索范围
resolve: {
alias: {
'@': resolve('src')
}
}
resolve: {
extensions: ['.vue', '.js']
}
happypack多线程执行
npm i happypack -D
const happyThreadPool = HappyPack.ThreadPool({ size: os.cpus().length });
module: {
rules: [
{
test: /\.js$/,
loader: 'happypack/loader?id=babel', // 原始loader替换成`happypack/loader`
include: [resolve('src')]
}
]
},
plugins: [
new HappyPack({
// id标识 需要处理的loader
id: 'babel',
// loader配置和原始配置一样
loaders: [
{
loader: 'babel-loader',
options: {
presets: ['es2015'],
cacheDirectory: true
}
}
],
threadPool: happyThreadPool
})
]
babel-plugin-dynamic-import-node异步加载
npm i babel-plugin-dynamic-import-node -D
"env": {
"development": {
"plugins": ["dynamic-import-node"]
},
"production": {
"plugins": ["dynamic-import-node"]
}
}
DllPlugin分包
const path = require('path');
const webpack = require('webpack');
const CleanWebpackPlugin = require("clean-webpack-plugin");
module.exports = {
entry: {
vendor: [
'vue',
'vue-router',
'vuex',
'axios',
'element-ui',
'echarts'
]
},
output: {
filename: '[name]_dll_[hash:6].js', // 产生的文件名
path: path.resolve(__dirname, '../static/dll'),
library: '[name]_dll_[hash:6]'
},
plugins: [
new CleanWebpackPlugin({
root: path.resolve(__dirname, '../static/dll'),
dry: false // 启用删除文件
}),
new webpack.DllPlugin({
name: '[name]_dll_[hash:6]',
path: path.resolve(__dirname, '../static/dll', '[name].dll.manifest.json')
})
]
};
const AddAssetHtmlPlugin = require('add-asset-html-webpack-plugin');
...
plugins: [
// 插入dll json
new webpack.DllReferencePlugin({
context: path.join(__dirname),
manifest: require('../static/dll/vendor.dll.manifest.json')
}),
new HtmlWebpackPlugin(),
// 插入 dll js
new AddAssetHtmlPlugin([{
publicPath: config.build.assetsPublicPath + 'static/dll/', // 注入到html中的路径
outputPath: 'static/dll/', // 输出文件目录
filepath: resolve('static/dll/*.js'), // 文件路径
includeSourcemap: false,
typeOfAsset: "js"
}])
]
减少 vendor 大小
module.exports = {
entry: {},
externals: {
'vue': vue
}
}
总结