js中common.js和ECMAScript.js区别对比分析

 更新时间:2025年05月29日 10:02:47   作者:以码平川  
这篇文章主要介绍了js中common.js和ECMAScript.js区别对比分析,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

以下是关于 CommonJS 和 ECMAScript Modules(ESM)的详细对比分析,包含底层原理和示例说明:

🧩 核心差异对比表

特性CommonJSES Modules
来源Node.js 社区规范ECMAScript 语言标准
加载方式动态加载(运行时解析)静态加载(编译时解析)
加载环境Node.js 原生支持浏览器原生支持,Node.js需开启 --experimental-modules(v13.2+已稳定)
语法格式require() / module.exportsimport / export
加载行为同步加载异步加载
模块解析文件路径需完整支持 bare module 说明符(需要导入映射)
变量访问修改原始导出对象绑定只读引用
循环引用处理部分加载(未完成的状态)引用预解析(存在TDZ)
顶层作用域模块内this指向module.exports顶层thisundefined
静态分析不支持 Tree-shaking支持 Tree-shaking 优化

📦 底层加载机制差异(图示)

CommonJS 运行时解析流程

1. 执行代码 → 2. 构建模块对象 → 3. 按需加载依赖 → 4. 包裹成函数执行

Module Wrapper 伪代码:

function (exports, require, module, __filename, __dirname) {
  // 用户代码在此执行
  module.exports = ...;
}

ESM 预解析流程

1. 解析阶段 → 2. 建立模块关系图 → 3. 编译阶段 → 4. 实例化 → 5. 执行代码

关键特性:

  • 模块记录(Module Record):存储导入/导出关系
  • 实时绑定(Live Bindings):导出值变化会同步到导入方

🛠️ 代码示例对比

模块导出差异

// CommonJS 动态修改
exports.a = 1;        // ⇨ { a: 1 }
module.exports = { b: 2 };  // 最终导出 { b: 2 }
// ESM 绑定不可变
export let count = 0;
export function increment() {
  count++;  // 所有导入模块都会看到更新后的值
}

循环依赖处理

// commonjs/a.js
console.log('a开始');
exports.done = false;
const b = require('./b');  // 此时b尚未完成加载
console.log('在a中,b.done =', b.done);
exports.done = true;
console.log('a结束');
// commonjs/b.js
console.log('b开始');
exports.done = false;
const a = require('./a');  // 此时a导出{done: false}
console.log('在b中,a.done =', a.done);
exports.done = true;
console.log('b结束');
# 执行结果:
a开始 → b开始 → 在b中,a.done = false → b结束 → 在a中,b.done = true → a结束

⚡ 现代项目中的互操作性

混合使用解决方案

// 在 ESM 中引入 CJS
import cjsModule from './commonjs-module.cjs';
// 在 CJS 中引入 ESM(需异步)
const esModule = await import('./es-module.mjs');

Package.json 配置

{
  "type": "module",       // 默认使用ESM
  "main": "./index.cjs",  // CJS入口
  "exports": {
    "import": "./esm/index.js",   // ESM入口
    "require": "./cjs/index.js"   // CJS入口
  }
}

🔧 转译工具处理原理(以Babel为例)

# 转换步骤示例
ESM → 解析为AST → 检测import/export → 替换为require语法 → 添加helper函数

示例转换效果:

// 原始ESM
import { readFile } from 'fs';
export const data = readFile('./file.txt');
// 转换后CommonJS
const { readFile } = require('fs');
exports.data = readFile('./file.txt'); 

🚀 性能优化差异

CommonJS 优化难点

  • 无法预知依赖关系,阻碍并行加载
  • 动态表达式导致死代码难以消除
require(condition ? 'a' : 'b'); // 无法静态分析

ESM 优化空间

// webpack利用静态分析实现的特性
import(/* webpackPrefetch: true */ './chart'); // 预取
import(/* webpackChunkName: "utils" */ './utils'); // 分块命名

🌐 浏览器支持情况

浏览器ESM支持版本
Chrome61+
Firefox60+
Safari10.1+
Edge16+
<!-- 浏览器直接使用ESM -->
<script type="module" src="app.js"></script>

💡 选用建议

Node.js 服务端

  • 新项目 > Node 14:优先使用ESM
  • 旧项目迁移:逐步替换关键模块

前端工程

  • 统一使用ESM(配合webpack等打包工具)
  • 第三方库需提供ESM版本(通过package.jsonmodule字段)

工具库开发

# 推荐双模式发布
lib/
├── esm/       # ESM版本(支持Tree-shaking)
├── cjs/       # CommonJS版本 
└── index.d.ts # 类型声明

两种模块系统在JavaScript生态中仍将长期共存,理解其底层机制有助于更高效地处理模块化问题。随着Node.js对ESM支持的完善,未来ESM会成为主流选择,但CommonJS仍将在老项目中持续存在。

到此这篇关于js中common.js和ECMAScript.js区别的文章就介绍到这了,更多相关js common.js和ECMAScript.js内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • JavaScript实现广告弹窗效果

    JavaScript实现广告弹窗效果

    这篇文章主要为大家详细介绍了JavaScript实现广告弹窗效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-08-08
  • JS循环中正确使用async、await的姿势分享

    JS循环中正确使用async、await的姿势分享

    async / await是ES7的重要特性之一,也是目前社区里公认的优秀异步解决方案,下面这篇文章主要给大家介绍了关于JS循环中正确使用async、await的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2021-12-12
  • JS与C#编码解码

    JS与C#编码解码

    这篇文章主要是对JS与C#编码解码进行了详细的介绍,需要的朋友可以过来参考下,希望对大家有所帮助
    2013-12-12
  • JS图像无缝滚动脚本非常好用

    JS图像无缝滚动脚本非常好用

    最好用的JS图像无缝滚动脚本,根据需要改改宽度和图像个数就可以了,需要的朋友可以参考下
    2014-02-02
  • 小程序简单两栏瀑布流效果的实现

    小程序简单两栏瀑布流效果的实现

    这篇文章主要介绍了小程序简单两栏瀑布流效果的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-12-12
  • JS未跨域操作iframe里的DOM

    JS未跨域操作iframe里的DOM

    这篇文章主要介绍了JS未跨域操作iframe里的DOM 的相关资料,非常不错具有参考借鉴价值,需要的朋友可以参考下
    2016-06-06
  • 前端canvas实现电子签约完成线上签署功能

    前端canvas实现电子签约完成线上签署功能

    最近发现现在租房还是签合同,越来越多采用电子签约的方式进行,好处不用多说节约成本,节约时间,下面我们来看看使用canvas实现电子签约完成线上签署功能
    2025-01-01
  • JS获取随机数和时间转换的简单实例

    JS获取随机数和时间转换的简单实例

    下面小编就为大家带来一篇JS获取随机数和时间转换的简单实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-07-07
  • 微信小程序自定义用户登录弹窗

    微信小程序自定义用户登录弹窗

    这篇文章主要为大家详细介绍了微信小程序自定义用户登录弹窗,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • JavaScripts数组里的对象排序的24个方法(最新整理收藏)

    JavaScripts数组里的对象排序的24个方法(最新整理收藏)

    文章介绍了24种在JavaScript中对数组进行排序的方法,每种方法都有具体的示例和应用场景,适合不同情况下的排序需求,感兴趣的朋友跟随小编一起看看吧
    2025-01-01

最新评论