前端工程化-Webpack 进阶-删除冗余代码、缓存

以下皆为拉勾教育课件内笔记

删除冗余代码(Tree Shaking)

简介

Tree Shaking 的作用是,打包时,删除未引用的代码(dead code)

何为未引用代码(dead code):

  • return 后的代码
  • 只声明,没有使用的代码
  • 只引入,没有使用的代码

因为 Webpack 的依赖是⼀个树形结构,因此,删除无用代码的过程称之为摇树(Tree Shaking)。例如:下图中的 函数 A 和 函数 D 是未引用的代码。Webpack 打包时,会将其“摇”掉。

前提

使用 ES Modules 规范的模块,才能执行 Tree Shaking。这是因为 Tree Shaking 依赖于 ES Modules 的静态语法分析(即代码未执行之前,通过 export 和 import 来分析,哪些代码未引用)

使用

  • 生产模式:Tree Shaking 自动开启
  • 开发模式:usedExports 、sideEffects

usedExports

  1. optimization.usedExports(标记没用的代码)
    /* unused harmony export xxxxx */
  2. terser-webpack-plugin(删除没用的代码)
    optimization.minimize: true(删除 unused harmony export xxxxx标记的代码 )
    Webpack 4 需要单独安装(Webpack 5 无需安装)
    详情查看:https://www.npmjs.com/package/terser-webpack-plugin
// webpack.config.js
// ...
const TerserPlugin = require("terser-webpack-plugin")

module.exports = (env, argv) => {
  // ...

  // 优化策略
    optimization: {
      // 标记未被使用的代码
      usedExports: true,
      // 删除 usedExports 标记的未使用的代码
      minimize: true,
      minimizer: [new TerserPlugin()]
    },

  // ...
}

Tree Shaking 与 Source Map 存在兼容性问题
想要使用 Tree Shaking,devtool 只能使用以下几种模式:
source-map | inline-source-map | hidden-source-map | nosources-source-map
eval 模式,将 JS 输出为字符串(不是 ES Modules 规范),导致 Tree Shaking 失效

sideEffects

sideEffects(副作用)

  • 无副作用:如果一个模块单纯的导入导出变量,那它就无副作用

  • 有副作用:如果一个模块还修改其他模块或者全局的一些东西,就有副作用

    • 修改全局变量
    • 在原型上扩展方法
    • CSS 的引入

sideEffects的作用:把未使用但无副作用的模块一并删除

对于没有副作用的模块,未使用代码不会被打包(相当于压缩了输出内容),如果模块或函数有副作用,则不能随便删除。如下图,删除有副作用的函数,可能影响正在使用的 Fun C

开启副作用(webpack.config.js)

  • optimization.sideEffects: true
// webpack.config.js
// ...

module.exports = (env, argv) => {
  // ...

  // 优化策略
    optimization: {
      sideEffects: true,
      //...
    },

  // ...
}

标识代码是否有副作用(package.json)

  • ''sideEffects''
    • false:所有代码都没有副作用(告诉 webpack 可以安全地删除未用的 exports)
    • true:所有代码都有副作用
    • 数组:(告诉 webpack 哪些模块有副作用,不删除)例如:['./src/wp.js', '*.css'],数组中可以使用相对路径,绝对路径或正则。
// package.json
"sideEffects": [
  './src/wp.js', 
  '*.css'
],

缓存

缓存中主要涉及两个内容

  • Babel 缓存
  • 文件资源缓存

Babel 缓存

cacheDirectory: true(第二次构建时,会读取之前的缓存) 如下图:二次构建时,可能只有 tab.js,此时,我们可以只构建 tab.js,其他内容使用首次构建的缓存。

// webpack.config.js
module.exports = (env, argv) => {
  // ...
  // 模块配置
  module: {
    rules: [
      // ...
      {
        test: /\.m?js$/,
        exclude: /node_modules/,
        use: {
          // 第二次构建时,会读取之前的缓存
          cacheDirectory: true,
          // ...
        }
      },
      // ...
    ]
  }
  // ...
}

文件资源缓存

如果代码在缓存期内,此时,代码更新了,如果还是继续访问缓存中的旧文件。就看不到实时效果。 方案:将代码文件名称,设置为哈希名称,名称发生变化时,就加载最新的内容。

Webpack 哈希值

  • [hash](每次 Webpack 打包生成的 hash 值)

  • [chunkhash](不同 chunk 的 hash 值不同 - 同一次打包可能生成不同的 chunk)

  • [contenthash](不同内容的 hash 值不同 - 同一个 chunk 中可能有不同的内容)

比如在出口配置的文件名称中添加哈希值,也可以在加:数字限定哈希位数。

module.exports = (env, argv) => {
  // ...

  // 出口配置
    output: {
      // 输出目录(输出目录必须是绝对路径)
      path: resolve(__dirname, 'output'),
      // 输出文件名称(8位哈希值文件名)
      filename: '[name].[contenthash:8].js'
    },

  // ...
}
[hash]

只要源码发生变化,打包后,所有的文件名称都会发生变化。

[chunkhash]

源码发生变化后,打包时,只有跟变化相关的一路打包,打包后的文件名才会发生变化。

[contenthash]

源码发生变化后,打包时,只有跟变化相关的一个文件的文件名会发生变化。

由此可知,上述三种不同类型的哈希,对缓存的控制力度不同。


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,271评论 5 466
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,725评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,252评论 0 328
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,634评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,549评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 47,985评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,471评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,128评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,257评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,233评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,235评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,940评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,528评论 3 302
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,623评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,858评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,245评论 2 344
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,790评论 2 339