Samgao0312 / Blog

MIT License
1 stars 1 forks source link

【积点成势】重学Webpack (5) —— 如何借助Webpack来优化性能 #152

Open Samgao0312 opened 2 years ago

Samgao0312 commented 2 years ago

作为一个项目的打包构建工具,在完成项目开发后经常需要利用 Webpack 对前端项目进行性能优化,常见的优化手段有如下几个方面:

1、JS代码压缩

terser是一个JavaScript的解释、绞肉机、压缩机的工具集,可以帮助我们压缩、丑化我们的代码,让bundle更小。在production模式下,webpack 默认就是使用 TerserPlugin 来处理我们的代码的。如果想要自定义配置它,配置方法如下。

const TerserPlugin = require('terser-webpack-plugin')
module.exports = {
    ...
    optimization: {
        minimize: true,
        minimizer: [
            new TerserPlugin({
                parallel: true              // 电脑cpu核数-1
            })
        ]
    }
}

TerserPlugin常用的属性如下:

2、CSS代码压缩

CSS压缩通常用于去除无用的空格等,不过因为很难去修改选择器、属性的名称、值等,所以我们可以使用另外一个插件:css-minimizer-webpack-plugin。配置如下:

const CssMinimizerPlugin = require('css-minimizer-webpack-plugin')
module.exports = {
    // ...
    optimization: {
        minimize: true,
        minimizer: [
            new CssMinimizerPlugin({
                parallel: true
            })
        ]
    }
}

3、Html文件代码压缩

使用 HtmlWebpackPlugin 插件来生成 HTML 的模板时候,可以通过配置属性 minify 进行 html 优化,配置如下。

module.exports = {
    ...
    plugin:[
        new HtmlwebpackPlugin({
            ...
            minify:{
                minifyCSS:false,   // 是否压缩css
                collapseWhitespace:false,   // 是否折叠空格
                removeComments:true   // 是否移除注释
            }
        })
    ]
}

4、文件大小压缩

对文件的大小进行压缩,可以有效减少http传输过程中宽带的损耗,文件压缩需要用到 compression-webpack-plugin插件,配置如下。

new ComepressionPlugin({
    test:/\.(css|js)$/,  // 哪些文件需要压缩
    threshold:500, // 设置文件多大开始压缩
    minRatio:0.7, // 至少压缩的比例
    algorithm:"gzip", // 采用的压缩算法
})

5、图片压缩

如果我们对bundle包进行分析,会发现图片等多媒体文件的大小是远远要比 js、css 文件要大的,所以图片压缩在打包方面也是很重要的。配置可以参考如下的方式:

module: {
  rules: [
    {
      test: /\.(png|jpg|gif)$/,
      use: [
        {
          loader: 'file-loader',
          options: {
            name: '[name]_[hash].[ext]',
            outputPath: 'images/',
          }
        },
        {
          loader: 'image-webpack-loader',
          options: {
            // 压缩 jpeg 的配置
            mozjpeg: {
              progressive: true,
              quality: 65
            },
            // 使用 imagemin**-optipng 压缩 png,enable: false 为关闭
            optipng: {
              enabled: false,
            },
            // 使用 imagemin-pngquant 压缩 png
            pngquant: {
              quality: '65-90',
              speed: 4
            },
            // 压缩 gif 的配置
            gifsicle: {
              interlaced: false,
            },
            // 开启 webp,会把 jpg 和 png 图片压缩为 webp 格式
            webp: {
              quality: 75
            }
          }
        }
      ]
    },
  ]
}

6、Tree Shaking

Tree Shaking 是一个术语,在计算机中表示消除死代码,依赖于ES Module的静态语法分析。在webpack实现Trss shaking有两种不同的方案: - usedExports:通过标记某些函数是否被使用,之后通过Terser来进行优化的 - sideEffects:跳过整个模块/文件,直接查看该文件是否有副作用

usedExports的配置方法很简单,只需要将usedExports设为true即可,如下。

module.exports = {
    ...
    optimization:{
        usedExports
    }
}

而sideEffects则用于告知webpack compiler在编译时哪些模块有副作用,配置方法是在package.json中设置sideEffects属性。如果sideEffects设置为false,就是告知webpack可以安全的删除未用到的exports,如果有些文件需要保留,可以设置为数组的形式。

"sideEffecis":[    "./src/util/format.js",    "*.css" // 所有的css文件]

7、代码分离

默认情况下,所有的JavaScript代码(业务代码、第三方依赖、暂时没有用到的模块)在首页全部都加载,就会影响首页的加载速度。如果可以分出出更小的bundle,以及控制资源加载优先级,从而优化加载性能。

代码分离可以通过splitChunksPlugin来实现,该插件webpack已经默认安装和集成,只需要配置即可。

module.exports = {   
 ...    
    optimization:{    
        splitChunks:{       
             chunks:"all"     
                }  
        }}

splitChunks有如下几个属性:

8、内联chunk

可以通过 InlineChunkHtmlPlugin 插件将一些 chunk 的模块内联到html,如runtime的代码(对模块进行解析、加载、模块信息相关的代码),代码量并不大但是必须加载的,比如:

const InlineChunkHtmlPlugin = require('react-dev-utils/InlineChunkHtmlPlugin')
const HtmlWebpackPlugin = require('html-webpack-plugin')
     module.exports = {  
           ...    plugin:[      
              new InlineChunkHtmlPlugin(HtmlWebpackPlugin,[/runtime.+\.js/]}

总结一下,Webpack对前端性能的优化,主要是通过文件体积大小入手,主要的措施有分包、减少Http请求次数等。