构建速度

1. 优化 babel-loader

  1. {
  2. test: /\.js$/,
  3. use: ['babel-loader?cacheDirectory'], // 开启缓存,未改变部分不重新编译
  4. include: path.resolve(__dirname, 'src') // 明确范围
  5. /*
  6. 排除范围,include和exclude两者选一个即可
  7. exclude: path.resolve(__dirname, 'node_modules')
  8. */
  9. }

2. webpack.IgnorePlugin 避免引入无用模块

例如 import moment from ‘moment’,默认会引入所有语言版本的js代码,代码过大
先是不使用IgnorePlugin

  1. // 引入moment,默认会引入全部语言包
  2. import moment from 'moment'
  3. moment.locale('zh-cn')
  4. console.log('local', moment.locale())
  5. console.log('date',moment().format('ll'))

打包查看index.[contentHash:8].js的尺寸,这个我关闭了代码分割功能
截屏2020-10-25 上午11.14.39.png
263kb,并且高亮警告⚠️了
我们在webpack.prod.js中配置一个新的plugin

  1. // webpack.pro.js
  2. plugins: [
  3. // 忽略 moment 下的 /locale 目录
  4. new webpack.IgnorePlugin(/\.\/locale/, /moment/),
  5. ]
  6. // 但是这么一忽略,moment的中文语言包也没了,于是需要修改index.js
  1. // index.js
  2. // 引入moment,举例IgnorePlugin
  3. import moment from 'moment'
  4. // 手动引入中文语言包
  5. import 'moment/locale/zh-cn'
  6. moment.locale('zh-cn')
  7. console.log('local', moment.locale())
  8. console.log('date',moment().format('ll'))

再次打包
截屏2020-10-25 上午11.22.28.png

3. noParse避免重复打包

像 react.min.js vue.min.js都是打包好的,文件就没有采用模块化

  1. module.exports = {
  2. module: {
  3. // 独立完整的 `vue.min.js` 文件就没有采用模块化
  4. // 忽略对 `react.min.js` 文件的递归解析处理
  5. noParse: [/vue\.min\.js$/]
  6. }
  7. }

IgnorePlugin 和 noParse 的区别

  • IgnorePlugin 直接不引入,代码中没有
  • noParse引入,但是不打包

    4. happyPack 多进程打包

  • JS单线程,开启多进程打包

  • 提高构建速度(特别是多核CPU) ```javascript // webpack.js // 这里babel-loader就不能放在common.js中了

const HappyPack = require(‘happypack’)

module.exports = { / …… / module: { rules: [ { test: /.js$/, // 把对 .js 文件的处理权交给 id 为 babel 的 HappyPack 实例 use: [‘happypack/loader?id=babel’], include: srcPath } ], / …… / plugins: [ / …… / new HappyPack({ // 用唯一标识 id 来代表当前的 HappyPack 是用来处理一类特定的文件 id: ‘babel’, // 如何处理 .js 文件,用法和 loader 配置一样 loaders: [‘babel-loader?cacheDirectory’] }) ] } }

  1. <a name="JejLT"></a>
  2. ## 5. ParallelUglifyPlugin多进程压缩JS
  3. - webpack内置Uglify工具压缩JS
  4. - JS单线程,开启多进程压缩更快
  5. - 和happyPack同理
  6. ```javascript
  7. // webpack.prod.js
  8. const ParallelUglifyPlugin = require('webpack-parallel-uglify-plugin')
  9. module.exports = {
  10. /* ...... */
  11. plugins: [
  12. new ParallelUglifyPlugin({
  13. uglifyJS: {
  14. output: {
  15. beautify: false, // 不美化输出,最紧凑的输出
  16. comment: false // 不保留注释,删除所有注释
  17. },
  18. compress: {
  19. // 删除所有consolelog语句
  20. drop_console: true,
  21. // 内嵌定义了但是只用到一次的变量
  22. collapse_vars: true,
  23. // 提取出出现多次但是还没有定义成变量去引用的静态值
  24. reduce_vars: true
  25. }
  26. }
  27. })
  28. ]
  29. }

关于开启多进程

  • 项目较大,打包较慢,开启多进程能提高速度
  • 项目较小,打包很快,开启多进程会降低速度(进程开销)
  • 按需使用

6. 自动刷新

一般不用自己配置,devServer会带上这个功能

  1. module.exports = {
  2. watch: true, // 开启监听,默认为false
  3. // 配置监听
  4. watchOptions: {
  5. ignored: /node_modules/, // 忽略哪些
  6. // 监听到变化后会等300ms再去执行动作,防止文件更新太快导致重新编译频率太高
  7. aggregateTimeout: 300, // 默认为 300ms
  8. poll: 1000 // 默认每隔1000毫秒询问一次
  9. }
  10. }

7. 热更新

自动刷新需要刷新整个网页,速度较慢,且状态会丢失。
热更新:新代码生效,网页不刷新,状态不丢失

  1. // webpack.dev.js
  2. const HotModuleReplacementPlugin = require('webpack/lib/HotModuleReplacementPlugin');
  3. moduel.exports = {
  4. mode: 'development',
  5. entry: {
  6. // 配置热更新需要更改的地方
  7. // index: path.join(srcPath, 'index.js')
  8. index: [
  9. 'webpack-dev-server/client?http://localhost:8080/',
  10. 'webpack/hot/dev-server',
  11. path.join(srcPath, 'index.js')
  12. ],
  13. other: path.join(srcPath, 'other.js')
  14. },
  15. /* ...... */
  16. plugins: [
  17. new HotModuleReplacementPlugin()
  18. ],
  19. devServer: {
  20. /* ...... */
  21. hot: true //开启热更新
  22. }
  23. }

配置热更新还需要主动配置运行热更新的模块

  1. // index.js
  2. import { sum } from './math'
  3. // 例如我们运行 math.js 文件支持热更新
  4. // 增加开启热更新之后的代码逻辑
  5. if(module.hot) {
  6. module.hot.accept(['./math'], () => {
  7. // 这里写模块热更新后的回调逻辑
  8. const sumRes = sum(10, 80)
  9. console.log('sumRes in hot', sumRes)
  10. })
  11. }

8. DllPlugin 动态链接库插件

  • 前端框架如 vue react,体积大,构建慢
  • 较稳定,不常升级版本
  • 同一个版本只构建一次即可,不用每次都重新构建

  • webpack已内置DllPlugin支持

  • DllPlugin 打包出dll文件
  • DllReferencePlugin 使用dll文件

第一步

配置打包dll文件的配置文件

  1. // webpack.dll.js
  2. const path = require('path')
  3. const DllPlugin = require('webpack/lib/DllPlugin')
  4. const { srcPath, distPath } = require('./paths')
  5. module.exports = {
  6. mode: 'development',
  7. // JS 执行入口文件
  8. entry: {
  9. // 把 React 相关模块的放到一个单独的动态链接库
  10. react: ['react', 'react-dom']
  11. },
  12. output: {
  13. // 输出的动态链接库的文件名称,[name] 代表当前动态链接库的名称,
  14. // 也就是 entry 中配置的 react 和 polyfill
  15. filename: '[name].dll.js',
  16. // 输出的文件都放到 dist 目录下
  17. path: distPath,
  18. // 存放动态链接库的全局变量名称,例如对应 react 来说就是 _dll_react
  19. // 之所以在前面加上 _dll_ 是为了防止全局变量冲突
  20. library: '_dll_[name]',
  21. },
  22. plugins: [
  23. // 接入 DllPlugin
  24. new DllPlugin({
  25. // 动态链接库的全局变量名称,需要和 output.library 中保持一致
  26. // 该字段的值也就是输出的 manifest.json 文件 中 name 字段的值
  27. // 例如 react.manifest.json 中就有 "name": "_dll_react"
  28. name: '_dll_[name]',
  29. // 描述动态链接库的 manifest.json 文件输出时的文件名称
  30. path: path.join(distPath, '[name].manifest.json'),
  31. }),
  32. ],
  33. }

第二步,运行打包

  1. npx webpack --config webpack.dll.js

我们会在dist目录下看到,react.dll.js 和 react.manifest.json 文件

第三步,使用

首先模板文件需要修改

  1. <!-- index.html -->
  2. <!DOCTYPE html>
  3. <html lang="en">
  4. <head>
  5. <meta charset="UTF-8">
  6. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7. <meta http-equiv="X-UA-Compatible" content="ie=edge">
  8. <title>Document</title>
  9. </head>
  10. <body>
  11. <div id="root"></div>
  12. 需要配置引入这个js文件
  13. <script src="./react.dll.js"></script>
  14. </body>
  15. </html>

在webpack.dev.js中

  1. const path = require('path')
  2. const webpack = require('webpack')
  3. const { smart } = require('webpack-merge')
  4. const webpackCommonConf = require('./webpack.common.js')
  5. const { srcPath, distPath } = require('./paths')
  6. // 第一,引入 DllReferencePlugin
  7. const DllReferencePlugin = require('webpack/lib/DllReferencePlugin');
  8. module.exports = smart(webpackCommonConf, {
  9. mode: 'development',
  10. module: {
  11. rules: [
  12. {
  13. test: /\.js$/,
  14. loader: ['babel-loader'],
  15. include: srcPath,
  16. exclude: /node_modules/ // 第二,不要再转换 node_modules 的代码
  17. },
  18. ]
  19. },
  20. plugins: [
  21. new webpack.DefinePlugin({
  22. // window.ENV = 'production'
  23. ENV: JSON.stringify('development')
  24. }),
  25. // 第三,告诉 Webpack 使用了哪些动态链接库
  26. new DllReferencePlugin({
  27. // 描述 react 动态链接库的文件内容
  28. manifest: require(path.join(distPath, 'react.manifest.json')),
  29. }),
  30. ],
  31. devServer: {
  32. port: 8080,
  33. progress: true, // 显示打包的进度条
  34. contentBase: distPath, // 根目录
  35. open: true, // 自动打开浏览器
  36. compress: true, // 启动 gzip 压缩
  37. // 设置代理
  38. proxy: {
  39. // 将本地 /api/xxx 代理到 localhost:3000/api/xxx
  40. '/api': 'http://localhost:3000',
  41. // 将本地 /api2/xxx 代理到 localhost:3000/xxx
  42. '/api2': {
  43. target: 'http://localhost:3000',
  44. pathRewrite: {
  45. '/api2': ''
  46. }
  47. }
  48. }
  49. }
  50. })