当前位置: 首页 > news >正文

前端模块化:CommonJS 与 ES Module

在现代 JavaScript 的运行时与构建工具中,模块系统已成为代码组织的核心机制。理解 CommonJS 与 ES Module 的设计哲学、加载策略以及语义差异,是构建可维护、可移植代码库的基础。

一、历史与定位

CommonJS 由社区提出,早期仅针对 Node.js 运行环境;ES Module 则在 ECMAScript 规范中标准化,旨在为浏览器与服务器提供统一的模块化语法。二者分别代表了“运行时 API”与“静态语法”两大范式。

二、模块解析策略

CommonJS 的 require 函数在运行时执行字符串路径解析,允许动态拼接模块标识符。模块代码在执行阶段被包装成一个立即执行函数,形参依次为 exports、require、module、__filename、__dirname。模块首次加载后,其 exports 对象被缓存,后续再次 require 直接返回缓存引用。
ES Module 则采用静态分析:import / export 语句在编译阶段即确定依赖关系。解析过程分为静态与动态两条路径:静态导入在语法层面完成绑定,动态 import() 返回 Promise,实现按需异步加载。

三、依赖执行模型

CommonJS 的依赖在执行阶段同步加载,模块内部代码自上而下顺序运行,require 出现即阻塞当前线程直至模块加载完成。
ES Module 的静态依赖在解析阶段完成链接,动态依赖通过异步 import 引入,不阻塞主线程。模块环境记录(Module Environment Record)确保所有导入标识符与导出标识符形成符号绑定,即导入变量与导出变量共享同一内存地址,运行时修改可双向感知。

四、导出语义

CommonJS 的 exports 与 module.exports 本质上是普通对象属性赋值,导出值在 require 调用时完成快照,后续变化不可见。

exports.a = 'a';
module.exports.b = 'b';
this.c = 'c';
module.exports = {d: 'd'
}

模块导出结果:

{ d: 'd' }

ES Module 的导出分为命名导出与默认导出。命名导出通过标识符绑定实现实时引用;默认导出绑定到模块命名空间对象的 default 属性,模块内部对导出值的修改会立即反映到导入方。

// module counter
var count = 1;
export {count}
export function increase(){count++;
}// module main
import { count, increase } from './counter';
import * as counter from './counter';
const { count: c } = counter;
increase();
console.log(count);
console.log(counter.count);
console.log(c);

输出结果:

2
2
2

五、常见陷阱

在 CommonJS 模块末尾重新赋值 module.exports 会完全覆盖之前的 exports 属性,导致先前附加到 exports 上的属性丢失。
在 ES Module 中,由于符号绑定,对导出变量的直接赋值会导致运行时错误,必须通过函数或对象属性间接修改。

六、实战建议

  1. 新项目优先采用 ES Module,利用静态分析与 Tree-Shaking 减少包体积。
  2. 需要运行时动态依赖解析的场景(插件系统、配置中心)可保留 CommonJS,或通过 dynamic import 桥接。
  3. 混用两种体系时,利用打包工具的 interop 机制统一模块格式,避免直接混用 require 与 import 导致的双重加载。
http://www.dtcms.com/a/294003.html

相关文章:

  • 性能测试-jmeter实战5
  • 4️⃣字典(dict)速查表
  • I2C控制器
  • 传统RNN模型笔记:输入数据长度变化的结构解析
  • 通用图片 OCR 到 Word API 数据接口
  • 数据结构自学Day13 -- 快速排序--“前后指针法”
  • 显微科研中的关键选择:不同显微镜相机技术特性与应用适配性全面解析
  • SpringCloudGateWay 使用nacos网关自动负载均衡
  • nrm指南
  • Sklearn 机器学习 线性回归
  • 解决sparksql创建出来的数据库路径错误的问题
  • Docker Desktop 打包Unity WebGL 程序,在Docker 中运行Unity WebGL 程序
  • Unity国际版下载链接分享(非c1国内版)
  • Java面试题035:一文深入了解Docker
  • SQL基础⑧ | 表格篇
  • Android插件化实现方案深度分析
  • 智能小e-外联系统文档
  • CentOS 7 Linux 用 yum 安装 Docker,含 Docker 镜像无法拉取问题(即 docker pull 失败)的解决方案
  • 基于LNMP分布式个人云存储
  • 深入理解 Java Builder 设计模式:解决构造函数爆炸问题
  • 旧系统UI焕新陷阱:保留业务习惯与引入新交互的平衡点把控
  • JNPF组织权限,让企业权限体系更清晰高效​
  • TARA分析学习
  • 亚远景-传统功能安全VS AI安全:ISO 8800填补的标准空白与实施难点
  • Django 入门详解:从零开始构建你的第一个 Web 应用
  • YOLOv5模型剪枝实战教程
  • 从8h到40min的极致并行优化:Spark小数据集UDTF处理的深度实践与原理剖析
  • 分布式系统中的缓存设计与应用
  • 前端实现可编辑脑图的方案
  • 一洽客服系统:小程序接入功能说明