WHAT - Tree Shaking 的前提是 ES Module

目录

  • [1. ESM 具有静态结构](#1. ESM 具有静态结构)
  • [2. CommonJS (CJS) 无法静态分析](#2. CommonJS (CJS) 无法静态分析)
  • [3. Tree-shaking 依赖静态分析](#3. Tree-shaking 依赖静态分析)
  • [4. 构建工具支持](#4. 构建工具支持)
  • 总结

WHAT - Webpack 详解系列(七) 中我们看到:

Tree-shaking 实现的前提是 ES Modules,也就是说:最终交给 Webpack 打包的代码,必须是使用 ES Modules 的方式来组织的模块化。

这是为什么?今天我们就来探讨一下。

Tree-shaking 是一种 消除未使用代码(dead code elimination) 的技术,是一个优化过程,具体同样可以先阅读 WHAT - Webpack 详解系列(七)。它的实现前提之一是 ES Modules (ESM),主要原因如下:

1. ESM 具有静态结构

  • ESM 在编译时即可确定依赖关系和导出内容 ,因为 importexport 语法是静态 的,不能在运行时动态修改:

    ts 复制代码
    import { foo } from "./module";
  • 这种静态分析 特性使得编译器(如 Webpack、Rollup、esbuild)可以在构建时直接识别未使用的导出,并进行优化。

ESM 示例(可 tree-shake):

ts 复制代码
// module.ts
export const foo = () => console.log("foo");
export const bar = () => console.log("bar");
ts 复制代码
// main.ts
import { foo } from "./module"; // 只引入了 foo
foo();
  • 这里 bar 没有被使用,构建工具可以安全地删除它。

2. CommonJS (CJS) 无法静态分析

  • CommonJS 使用 require()导入是动态的 ,在运行时才会解析:

    js 复制代码
    const module = require("./module");
    console.log(module.foo);
  • 由于 module 是一个对象,它的属性(foobar)可能在运行时被修改或动态访问,无法在构建时确定哪些代码未被使用

CJS 示例(无法 tree-shake):

js 复制代码
// module.js
exports.foo = () => console.log("foo");
exports.bar = () => console.log("bar");
js 复制代码
// main.js
const module = require("./module");
module.foo(); // 可能在运行时动态访问 bar,无法优化
  • 由于 module 是对象,module.bar 可能在运行时被访问,因此无法安全删除 bar

3. Tree-shaking 依赖静态分析

  • ESM 的静态结构 使得构建工具可以直接在代码构建阶段分析哪些模块、函数、变量未被使用,并进行摇树优化(Tree-shaking)
  • CJS 是动态的 ,构建工具无法可靠确定哪些代码未被使用,因此无法进行 Tree-shaking。

4. 构建工具支持

  • Rollup(默认支持 ESM,Tree-shaking 效果最好)
  • Webpack (支持 ESM,但需要 mode: 'production'optimization.usedExports
  • esbuild、Vite(基于 ESM,Tree-shaking 效果优秀)

总结

ES Modules (ESM) CommonJS (CJS)
导入方式 import { foo } from "./module" const module = require("./module")
是否静态 ✅ 静态,编译时可分析 ❌ 动态,运行时才确定
Tree-shaking 支持 ✅ 支持(构建时可优化未使用的代码) ❌ 不支持(构建时无法安全删除代码)
推荐使用 ✅ 推荐(现代前端最佳实践) 🚫 仅用于 Node.js 兼容场景

最佳实践

  • 前端项目(React, Vue, Svelte, etc.)使用 ESM,配合 Webpack/Vite/rollup 进行 Tree-shaking。
  • Node.js 项目 ➝ 如果可以,使用 "type": "module" 迁移到 ESM;如果必须用 CJS,Tree-shaking 就不可行了。

如果你的目标是优化打包体积,请优先使用 ESM,这样构建工具才能高效地进行 Tree-shaking!

相关推荐
m0_471199633 分钟前
【自动化】前端开发,如何将 Jenkins 与 Gitee 结合实现自动化的持续集成(构建)和持续部署(发布)
前端·gitee·自动化·jenkins
w***95494 分钟前
spring-boot-starter和spring-boot-starter-web的关联
前端
Moment8 分钟前
富文本编辑器技术选型,到底是 Prosemirror 还是 Tiptap 好 ❓❓❓
前端·javascript·面试
xkxnq13 分钟前
第二阶段:Vue 组件化开发(第 18天)
前端·javascript·vue.js
晓得迷路了14 分钟前
栗子前端技术周刊第 112 期 - Rspack 1.7、2025 JS 新星榜单、HTML 状态调查...
前端·javascript·html
怕浪猫17 分钟前
React从入门到出门 第五章 React Router 配置与原理初探
前端·javascript·react.js
jinmo_C++17 分钟前
从零开始学前端 · HTML 基础篇(一):认识 HTML 与页面结构
前端·html·状态模式
鹏多多23 分钟前
前端2025年终总结:借着AI做大做强再创辉煌
前端·javascript
小Tomkk32 分钟前
⭐️ StarRocks Web 使用介绍与实战指南
前端·ffmpeg
不一样的少年_36 分钟前
产品催: 1 天优化 Vue 官网 SEO?我用这个插件半天搞定(不重构 Nuxt)
前端·javascript·vue.js