普通视图

发现新文章,点击刷新页面。
昨天 — 2026年5月4日首页

ESModule和Commonjs模块的区别

作者 Rkgua
2026年5月3日 22:12

ES Module(ESM)和 CommonJS(CJS)是 JavaScript 中两种主流的模块化规范。ESM 是 ES6 推出的官方标准,而 CommonJS 则是 Node.js 早期采用的模块化方案。

以下从几个核心角度为你详细拆解:

1. 核心差异速览表

对比角度 CommonJS (CJS) ES Module (ESM)
基本语法 require() 导入,module.exports 导出 import 导入,export 导出
加载时机 运行时加载(动态) 编译时加载(静态)
加载方式 同步加载 异步加载(浏览器端)
导出本质 值的拷贝(浅拷贝) 值的引用(Live Binding)
代码优化 不支持 Tree Shaking 支持 Tree Shaking
顶层 this 指向 module.exports undefined(严格模式)

2. 深度解析各个角度

语法与规范来源

  • CommonJS:是社区提出的规范,主要用于 Node.js 服务端环境。它的语法非常直观,使用 require() 来引入模块,使用 module.exportsexports 来向外暴露功能。
  • ES Module:是 ECMAScript 2015 (ES6) 的官方语言标准,旨在统一浏览器和服务端的模块化。它使用 importexport 关键字,语法更加语义化,支持命名导出和默认导出。

加载时机与方式(最核心的区别)

  • CommonJS 是“运行时同步加载”:当你代码执行到 require() 这一行时,才会去加载并执行对应的模块文件。这种方式在服务端(读取本地硬盘文件)非常高效,但在浏览器端会因为网络请求阻塞页面渲染,所以浏览器不原生支持。
  • ES Module 是“编译时静态加载”:JS 引擎在解析代码的阶段(编译时),就会通过分析 importexport 语句,提前确定好模块之间的依赖关系。在浏览器中,ESM 默认是异步加载的,不会阻塞 HTML 的解析。

导出的本质:值拷贝 vs 值的引用 这是两者在实际开发中最容易产生 Bug 的差异点:

  • CommonJS(值拷贝):导出的是模块内部变量的一个副本。如果模块内部修改了这个变量,外部引入的地方是感知不到的。
    // CommonJS 示例
    // counter.js
    let count = 0;
    module.exports = { count };
    setTimeout(() => { count = 1; }, 1000); // 内部修改
    
    // main.js
    const { count } = require('./counter.js');
    console.log(count); // 0
    setTimeout(() => { console.log(count); }, 1100); // 依然是 0,因为是拷贝的旧值
    
  • ES Module(值的引用 / Live Binding):导出的是对模块内部变量的动态引用。当模块内部修改了变量,所有引入该变量的地方都会同步更新。
    // ESM 示例
    // counter.js
    export let count = 0;
    setTimeout(() => { count = 1; }, 1000); // 内部修改
    
    // main.js
    import { count } from './counter.js';
    console.log(count); // 0
    setTimeout(() => { console.log(count); }, 1100); // 1,实时同步了最新值
    

代码优化(Tree Shaking)

  • ES Module:由于它是静态的,打包工具(如 Webpack、Rollup、Vite)可以在打包阶段就分析出哪些代码被使用了,哪些没有。未被使用的代码(Dead Code)会被直接剔除,这个过程叫 Tree Shaking(摇树优化),能显著减小打包体积。
  • CommonJS:由于 require() 可以在代码运行时动态执行(比如写在 if 判断里),打包工具很难在编译阶段确定到底引用了哪些模块,因此无法有效支持 Tree Shaking。

运行环境与兼容性

  • CommonJS:Node.js 的默认模块规范,生态极其成熟。在浏览器中无法直接使用,必须通过 Webpack、Browserify 等工具打包转换。
  • ES Module:现代浏览器原生支持(通过 <script type="module">),也是现代前端框架(Vue3, React)和构建工具(Vite)的首选。Node.js 从 v12 版本后也开始支持 ESM,但需要在 package.json 中配置 "type": "module" 或使用 .mjs 后缀。

总结建议: 在现代前端开发和新的 Node.js 项目中,优先推荐使用 ES Module,因为它更标准、性能更好且支持代码优化。但在维护一些老旧的 Node.js 项目或依赖某些仅支持 CJS 的第三方库时,你依然会频繁接触到 CommonJS。

事件流模型是什么和DOM事件模型等关系

作者 Rkgua
2026年5月2日 21:03

(对事件流和DOM,BOM误解,所发出的疑惑) 问题涉及 JavaScript 事件机制的核心概念。我们来系统梳理:

“事件流模型”是什么?它和‘浏览器事件模型’、‘文档事件模型’有什么关系或区别?”

实际上,“浏览器事件模型”和“文档事件模型”并不是标准术语,它们很可能是对以下两个概念的模糊表述:

  • DOM 事件模型(Document Object Model Event Model):即 W3C 标准定义的事件处理规范。
  • 浏览器对事件流的具体实现:所有现代浏览器都遵循 W3C DOM 事件模型。

因此,更准确地说,我们应该讨论的是:


一、什么是 事件流模型(Event Flow Model)

事件流描述的是:当一个事件(如点击)发生时,浏览器如何在 DOM 树中传播这个事件的顺序

✅ W3C 标准定义的事件流包含 三个阶段

  1. 捕获阶段(Capturing Phase)

    • 事件从 windowdocumenthtmlbody → ... → 目标元素的父级
    • 目的:允许祖先元素“提前拦截”事件
  2. 目标阶段(Target Phase)

    • 事件到达实际触发的元素(即 e.target
    • 此时既可视为捕获结束,也可视为冒泡开始
  3. 冒泡阶段(Bubbling Phase)

    • 事件从目标元素 → 父元素 → ... → bodyhtmldocumentwindow
    • 这是最常用的阶段,事件委托就依赖它

🌰 举例:点击一个 <button>,事件会先“下潜”到 button(捕获),然后在 button 上触发(目标),再“上浮”回页面顶部(冒泡)。


二、什么是 DOM 事件模型(Event Model)

“事件模型”指的是 如何为元素绑定事件监听器的机制,历史上有三种主要模型:

模型 名称 特点 是否支持事件流
DOM0 级 原始模型 element.onclick = fn只能绑定一个处理函数简单、兼容性好 ❌ 不支持捕获✅ 部分浏览器以冒泡方式实现
IE 事件模型 IE 特有(已淘汰) attachEvent()只支持冒泡 ✅ 仅冒泡
DOM2 级 现代标准 addEventListener(type, handler, useCapture)可绑定多个监听器支持捕获/冒泡 ✅ 完整支持三阶段事件流

🔍 关键点

  • 事件流 是“事件如何传播”的过程
  • 事件模型 是“如何注册监听器”的接口规范
  • DOM2 级事件模型 实现了 完整的事件流

三、“浏览器事件模型” vs “文档事件模型”?——澄清误解

这两个说法并非官方术语,但可以这样理解:

说法 实际指代 说明
浏览器事件模型 浏览器对 W3C DOM 事件标准的实现 所有现代浏览器(Chrome、Firefox、Safari 等)都遵循 W3C DOM Level 2/3 Events 规范
文档事件模型 DOM(文档对象模型)中的事件处理机制 即 W3C 定义的 EventTarget.addEventListener 等 API

结论

它们本质上是同一套东西 —— 浏览器实现了 W3C 定义的 DOM 事件模型,而该模型规定了 事件流 的行为。


四、如何控制事件流?

虽然不能“关闭”事件流,但可以干预:

1. 在捕获 or 冒泡阶段监听

// 捕获阶段
parent.addEventListener("click", handler, true);

// 冒泡阶段(默认)
child.addEventListener("click", handler, false); // 或省略第三个参数

2. 阻止传播

e.stopPropagation(); // 阻止继续冒泡或捕获
e.stopImmediatePropagation(); // 还阻止同元素其他监听器执行

3. 判断当前处于哪个阶段

e.eventPhase === Event.CAPTURING_PHASE; // 1
e.eventPhase === Event.AT_TARGET; // 2
e.eventPhase === Event.BUBBLING_PHASE; // 3

五、总结:核心关系图

W3C DOM 事件标准
│
├── 定义了「事件模型」→ 如何绑定监听器(addEventListener)
│
└── 定义了「事件流」→ 事件传播的三阶段(捕获 → 目标 → 冒泡)
     │
     └── 浏览器(Chrome/Firefox等)实现了这一标准
          → 所谓“浏览器事件模型”就是 DOM 事件模型的实现

💡 记住

  • e.target 始终是最初触发事件的元素
  • e.currentTarget当前执行回调的监听器所绑定的元素
  • 默认监听的是冒泡阶段,这是事件委托的基础

掌握这些,你就真正理解了 JS 事件的底层逻辑!

❌
❌