性能优化篇之Webpack构建代码质量压缩的建议

Webpack构建速度优化基本优化完毕,接下来考虑的就是:线上代码质量的优化,即如何使用webpack构建出高质量的代码

Webpack构建流程:初始化配置参数 -> 绑定事件钩子回调 -> 确定Entry逐一遍历 -> 使用loader编译文件 -> 输出文件

提纲

本次优化构建代码质量基本技术:

  • reactRouter按需加载;
  • 公共代码提取,以及代码压缩;
  • CDN接入;
  • 开启gzip压缩;
  • 接入treeShaking,剔除无用代码
  • 开启Scope Hoisting

(生产环境代码构建)为实时查看每次配置后代码构建情况,使用Webpack监听文件避免每次手动build,并且开启webpack-jarvis,实时查看构建分析,npm i -D webpack-jarvis

开启监听模式

watch: true,
watchOptions: {
  ignored: /node_modules/, // 忽略监听文件
  aggregateTimeout: 300, //文件变动后多久发起构建
  poll: 1000, //每秒询问次数,越小越好
}

一、react-router4实现按需加载

单页应用按需加载一般原则:

  1. 将网站划分成一个个小功能,在按照每个功能的相关度将他们分成几个类;
  2. 将没一个类合并成一个chunk,按需加载对应的代码;
  3. 不可将用户首次进入网站时需要看到画面的对应功能Chunk按需加载;

被分割出去的代码的加载需要一定的触发时机,即当用户操作了或者即将操作对应功能时再去加载对应的代码(默认使用react-router按需加载的触发条件是路由的变化)

实现条件:

使用插件:npm i react-loadable;

配合bable插件npm i @babel/plugin-syntax-dynamic-import;

代码示例:

// .bablerc
{
 "plugins": ["@babel/plugin-syntax-dynamic-import"]
}

// 示例代码
Loadable({
 loader: () => import('./component'), //按需加载组件
 loading: Loading, //处理组件加载的loading、error等
 delay: 300 //延迟加载避免loading的闪烁问题
});

// Loading组件自定义
// 接受三个props,其中pastDelay:等待时触发;timedOut:超时时触发超过delay;error:出错触发默认为200ms
const Loading = ({ pastDelay, timedOut, error }) => {
  if (pastDelay) {
   return <Spin spinning tip="Loadding..." ><div style={{height: 300}} /></Spin>
  } else if (timedOut) {
   return <Spin spinning tip="Taking a long time..." ><div style={{height: 300}} /></Spin>
  } else if (error) {
   return <div>Error!</div>;
  }
  return null;
};

二、提取公共代码webpack.optimization

optimization: {
  splitChunks: {
   chunks: "all",
   cacheGroups: {
    vendors: {
      test: /node_modules/,
      name: 'vendors',
      minSize: 0,
      minChunks: 1,
      chunks: 'initial',
      priority: 2 // 该配置项是设置处理的优先级,数值越大越优先处理
    },
    commons: {
     name: "comomns",
     test: resolve("src/components"), // 可自定义拓展规则
     minChunks: 2, // 最小共用次数
     minSize:0,  //代码最小多大,进行抽离
     priority: 1, //该配置项是设置处理的优先级,数值越大越优先处理
    }
  }
}

三、压缩文件js\css

使用npm i -D webpack-parallel-uglify-plugin启用多线程并行压缩JS

optimization: {
  minimizer: [
    new ParallelUglifyPlugin({
      cacheDir: '.cache/', //缓存压缩,默认不缓存,设置存放位置开启
      test: /.js$/, //匹配需要压缩的文件,默认为/.js$/和Loader配置一样
      //include: [], 使用正则去选择需要被压缩的文件和Loader配置一样
      //exclude: [], 使用正则去去除不需要被压缩的文件和Loader配置一样
      //workerCount: 2, 开启几个子进程并发执行压缩
      // sourceMap: false, 是否输出source Map,开启会导致压缩变慢
      // uglifyJS: {}, 用于压缩ES6代码不可和uglifyJS同时使用
      uglifyJS:{//压缩ES5代码
        output: {
          // 是否输出可读性较强的代码,即会保留空格和制表符,默认为输出,为了达到更好的压缩效果,可以设置为false
          beautify: false,
          //是否保留代码中的注释,默认为保留,为了达到更好的压缩效果,可以设置为false
          comments: false
        },
        compress: {
          //是否在UglifyJS删除没有用到的代码时输出警告信息,默认为输出
          warnings: false,
          //是否删除代码中所有的console语句,默认为不删除,开启后,会删除所有的console语句
          drop_console: true,
          //是否内嵌虽然已经定义了,但是只用到一次的变量,比如将 var x = 1; y = x, 转换成 y = 1, 默认为否
          collapse_vars: true,
          // 提取出现多次但是没有定义成变量去引用的静态值
          reduce_vars:true
        }
      },
    }),
  ]
},

提取和压缩Css

使用插件:optimize-css-assets-webpack-pluginmini-css-extract-plugin

使用示例:

// 提取css到单独的文件
const MiniCssExtractPlugin = require("mini-css-extract-plugin");
// optimizeCssPlugin CSS文件压缩插件
const optimizeCssPlugin = require('optimize-css-assets-webpack-plugin');

const extractSCSS = new MiniCssExtractPlugin({
  filename: 'css/[name].[contenthash:8].css',
  chunkFilename: 'css/[name]_[contenthash:8].css',
  fallback:'style-loader'
});
...
...
plugins: [
  new optimizeCssPlugin({
    assetNameRegExp: /\.css$/g,
    cssProcessor: require('cssnano'),
    cssProcessorOptions: { discardComments: { removeAll: true } },
    canPrint: true
  }),
]

webpack配置接入CDN

CDN

网站接入CDN,需要将网页的静态资源上传到CDN服务器,使用CDN地址访问;

使用CDN可以决解资源并行下载限制,处理静态资源Cookie同域名携带等问题;

CDN缓存和回源需要合理的设置静态资源hash接入CDN会引入多个域名,增加域名解析时间,可进行预解析域名<link rel="dns-prefetch" href="//js.dns.com" rel="external nofollow" />

webpack实现接入

  • output.publicPath设置JavaScript地址
  • css-loader.publicPath设置CSS导入的资源地址
  • WebPlugin.stylePublicPath中设置Css文件地址
// JavaScript
output: {
  publicPath: '//js.cdn.com/js/',
  path: path.join(__dirname, '../docs/dist'), // 打包后的文件存放的地方
  // 为输出的JavaScript文件名加上Hash值使用`chunkhash`(chunkhash:根据模块内容变化;hash: 根据每次构建随机)
  filename: "js/[name].[chunkhash:8].js",
  chunkFilename: "js/[name]-[id].[chunkhash:8].js",
},

开启gzip压缩

使用插件:npm i -D compression-webpack-plugin;

webpack配置

const CompressionPlugin = require("compression-webpack-plugin");

plugins: [
  new CompressionPlugin({
    filename: '[path].gz[query]', //目标资源名称。[file] 会被替换成原资源。[path] 会被替换成原资源路径,[query] 替换成原查询字符串
    algorithm: 'gzip',//算法
    test: /\.(js|css)$/,  //压缩 js 与 css
    threshold: 10240,//只处理比这个值大的资源。按字节计算
    minRatio: 0.8//只有压缩率比这个值小的资源才会被处理
  })
]

后台开启使用koa

const staticCache = require('koa-static-cache');
import config from './configs';

const app = new Koa();

app.use(staticCache(path.resolve(__dirname, "../dist"), {
  maxAge: 7 * 24 * 60 * 60,
  gzip: true, //开启
  dynamic: true,
}))

接入treeShaking,剔除无用代码

Tree Shaking可以用来找出有用代码,去除JavaScript中用不上的死代码;但是它依赖于ES6静态花模块语法import\export的导入和导出

webpack接入

修改.babelrc保留ES6模块话语句

注意新版本babel-preset-env已经预设babel-preset-es2015,babel推荐使用babel-preset-env取代babel-preset-es2015,并且继续使用babel-preset-es2015会发出警告信息。

{
 "presets": [
  ["env", {
   "modules": false
  }]
 ],
 "plugins": ["syntax-dynamic-import"]
}

webpack --display-used-exports运行构建带上--display-used-exports可追踪到Tree Shaking的工作;

Webpack只能正确的分析出如何剔除死代码,需要接入UglifyJs处理剔除(配置见上)

开启Scope Hoistion

scope hoisting即作用域提升;

在构建过程中,webpack会借助ES6 模块化的静态特性,确定模块的依赖关系,将一个bundle中的静态依赖提升到顶部。(所以需要和接入treeShaking一样配置Babel开启ES6模块化)

原理:分析模块间的依赖关系,尽可能的将零散的模块合并到一个函数中去,前提不能造成代码冗余,因此只有被引用了一次的模块才能被合并。

接入好处:

代码体积减少

代码在运行时因为创建的函数作用域更少了,内存开销也随之变小

webpack接入ModuleConcatenationPlugin内置插件

const ModuleConcatPlugin = require('webpack/lib/optimize/ModuleConcatenationPlugin');
plugins: [
   new ModuleConcatPlugin(), //开启scope Hoisting
 ],

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 性能优化篇之Webpack构建速度优化的建议

    如何输出Webpack构建分析 输出Webpack构建信息的.json文件:webpack --profile --json > stats.json --profile:记录构建中的耗时信息 --json:以json格式输出构建结果,最后只输出一个json文件(包含所有的构建信息) web可视化查看构建分析:得到了webpack构建信息文件stats.json,如何进行很好的可视化查看? 方案一:通过可视化分析工具Webpack Analyse,是个在线Web应用,上传stats.json文件

  • 详解webpack性能优化——DLL

    Webpack性能优化的方式有很多种,本文之所以将 dll 单独讲解,是因为 dll 是一种最简单粗暴并且极其有效的优化方式. 在通常的打包过程中,你所引用的诸如:jquery.bootstrap.react.react-router.redux.antd.vue.vue-router.vuex 等等众多库也会被打包进 bundle 文件中.由于这些库的内容基本不会发生改变,每次打包加入它们无疑是一种巨大的性能浪费. Dll 的技术就是在第一次时将所有引入的库打包成一个 dll.js 的文件,将

  • webpack学习教程之前端性能优化总结

    前言 曾几何时,我们是如上图的方式引入JS资源的,相信现在很少遇见了.近年来Web前端开发领域朝着规范开发的方向演进.体现在以下两点: 1.MVC研发构架.多多益处(逻辑清晰,程序注重数据与表现分离,可读性强,利于规避和排查问题...) 2.构建工具层出不穷.多多益处(提升团队协作,以及工程运维,避免人工处理琐碎而重复的工作) 模块化开发 将前端性能优化理论落地,代码压缩,合并,缓存控制,提取公共代码等 其他的还包括比如你可以用ES 6 或CoffeeScript写源码,然后构建出浏览器支持的E

  • 浅谈webpack性能榨汁机(打包速度优化)

    最近对项目的本地开发环境进行了打包速度优化,原有项目,网上能搜到的优化方案基本都加了,在16年低配mac pro 上打包时间为25秒多,但我发现细节做一些调整可能大大降低打包时间,最终优化到7秒多 dll 原有项目是线上和本地公用一套dll配置,因为antd这类ui库需要按需加载所以不能放到dll中,这时可以单独写一个dll配置,将所有第三方库添加到dll中. 这时因为.babelrc中添加了babel-plugin-import插件会导致优化不生效,所以需要对开发环境单独配置babel opt

  • Webpack性能优化 DLL 用法详解

    前言 在用 Webpack 打包的时候,对于一些不经常更新的第三方库,比如 react,lodash,我们希望能和自己的代码分离开,Webpack 社区有两种方案 CommonsChunkPlugin DLLPlugin 对于 CommonsChunkPlugin,webpack 每次打包实际还是需要去处理这些第三方库,只是打包完之后,能把第三方库和我们自己的代码分开.而DLLPlugin 则是能把第三方代码完全分离开,即每次只打包项目自身的代码. 用法 要使用 DLLPlugin,需要额外新建

  • 浅谈webpack 构建性能优化策略小结

    背景 如今前端工程化的概念早已经深入人心,选择一款合适的编译和资源管理工具已经成为了所有前端工程中的标配,而在诸多的构建工具中,webpack以其丰富的功能和灵活的配置而深受业内吹捧,逐步取代了grunt和gulp成为大多数前端工程实践中的首选,React,Vue,Angular等诸多知名项目也都相继选用其作为官方构建工具,极受业内追捧.但是,随者工程开发的复杂程度和代码规模不断地增加,webpack暴露出来的各种性能问题也愈发明显,极大的影响着开发过程中的体验. 问题归纳 历经了多个web项目

  • 性能优化篇之Webpack构建代码质量压缩的建议

    Webpack构建速度优化基本优化完毕,接下来考虑的就是:线上代码质量的优化,即如何使用webpack构建出高质量的代码 Webpack构建流程:初始化配置参数 -> 绑定事件钩子回调 -> 确定Entry逐一遍历 -> 使用loader编译文件 -> 输出文件 提纲 本次优化构建代码质量基本技术: reactRouter按需加载: 公共代码提取,以及代码压缩: CDN接入: 开启gzip压缩: 接入treeShaking,剔除无用代码 开启Scope Hoisting (生产环境

  • webpack构建打包的性能优化实战指南

    目录 前言 一.优化打包构建速度,提升开发体验和效率 1.1优化babel-loader 1.2IgnorePlugin,避免引入无用模块 1.3noParse避免重复模块化解析 1.4happyPack多进程打包 1.5ParallelUglifyPlugin多进程压缩js 1.6热更新 1.7DllPlugin动态链接库插件 二.webpack性能优化-产出代码 总结 前言 开发的时候,如果每次我们修改了文件,webpack都能很迅速地帮我们编译完构建完而且浏览器能保存状态更新内容,体验会比

  • webpack 如何同时输出压缩和未压缩的文件的实现步骤

    有的时候我们想要同时生成压缩和未压缩的文件,比如我们构建 lib 包的时候,我们希望用户能够使用压缩过后的代码文件作为 cdn 文件,最简单的一个方式就是通过指定环境变量,比如指定 MINIFY,如下: const path = require('path') const isMinify = process.env.MINIFY /** * @type {import('webpack').Configuration} */ const config = { entry: { index: '

  • 浅谈React + Webpack 构建打包优化

    本文介绍了React + Webpack 构建打包优化,分享给大家,具体如下: 使用 babel-react-optimize对 React 代码进行优化 检查没有使用的库,去除 import 引用 按需打包所用的类库,比如 lodash . echart 等 lodash 可以采用babel-plugin-lodash进行优化. 需要注意的是 在 babel-react-optimize 中使用了 babel-plugin-transform-react-remove-prop-types 这

  • 详解组件库的webpack构建速度优化

    背景 在公司的主要工作是组件库(基于vue的ui组件库,类似element-ui)的开发,也已经有两个多月,期间一直觉得项目的开发构建太慢,每次开发打开开发环境需要 40s 左右,简直不能忍.前前后后尝试了各种优化手段,但是都不理想.终于在今天,找到了问题所在,构建速度提升了 50% 以上,现在只需要 17s 左右,整个心情都好了.现在记录一下所用到的各种优化手段,因为是开发环境,所以只考虑构建速度. 各种配置项的优化 主要是对一些loader添加 include exclude之类的小优化,其

  • Android性能优化系列篇UI优化

    目录 前言 一.UI优化 1.1 系统做的优化 1.1.1 硬件加速 1.2 优化方案 1.2.1 java代码布局 1.2.2 View重用 1.2.3 异步创建view 1.2.4 xml布局优化 1.2.5 异步布局框架Litho 1.2.6 屏幕适配 1.2.7 Flutter 1.2.8 Jetpack Compose 1.3 工具篇 1.3.1 Choreographer 1.3.2 LayoutInspector/Android Device Monitor 1.3.3 Systr

  • Python 代码性能优化技巧分享

    如何进行 Python 性能优化,是本文探讨的主要问题.本文会涉及常见的代码优化方法,性能优化工具的使用以及如何诊断代码的性能瓶颈等内容,希望可以给 Python 开发人员一定的参考. Python 代码优化常见技巧 代码优化能够让程序运行更快,它是在不改变程序运行结果的情况下使得程序的运行效率更高,根据 80/20 原则,实现程序的重构.优化.扩展以及文档相关的事情通常需要消耗 80% 的工作量.优化通常包含两方面的内容:减小代码的体积,提高代码的运行效率. 改进算法,选择合适的数据结构 一个

随机推荐