说说模块化规范?CommonJS和ES Module的区别?

一、常见模块化规范

1. CommonJS(CJS)

  • 主要用于 Node.js

  • 同步加载模块

  • 通过 requiremodule.exports / exports 实现

    // a.js
    module.exports = {
    name: 'test'
    }

    // b.js
    const a = require('./a')
    console.log(a.name)


2. ES Module(ESM)

  • ES6 官方标准(浏览器 & Node 都支持)

  • 使用 import / export

  • 支持静态分析(这是核心优势)

    // a.js
    export const name = 'test'

    // b.js
    import { name } from './a.js'


3. AMD(Asynchronous Module Definition)

  • 浏览器端早期方案(代表:RequireJS)

  • 异步加载

    define(['a'], function(a) {
    return a
    })


4. CMD(Common Module Definition)

  • 阿里提出(代表:SeaJS)

  • 按需加载(依赖就近)

    define(function(require, exports) {
    const a = require('./a')
    })


二、CommonJS vs ES Module 核心区别

这是面试高频重点,我帮你总结成"7大差异"👇


1️⃣ 加载机制(最重要)

CommonJS ES Module
加载方式 运行时加载 编译时加载(静态分析)
本质 动态 require 静态 import

👉 解释:

  • CommonJS:代码执行时才加载

  • ESM:编译阶段就确定依赖关系(Tree Shaking 依赖这个)


2️⃣ 是否支持 Tree Shaking

  • CommonJS ❌ 不支持

  • ES Module ✅ 支持

👉 因为 ESM 是静态结构


3️⃣ 导出值的机制(重点)

CommonJS ES Module
导出 值的拷贝 引用绑定(live binding)
示例:

CommonJS:

复制代码
let count = 0
module.exports = { count }
count++

👉 引入后是旧值


ES Module:

复制代码
export let count = 0
count++

👉 引入后是最新值(实时绑定)


4️⃣ 是否可以动态加载

CommonJS ES Module
动态加载 ✅ require ✅ import()

👉 但:

  • require 是同步

  • import() 是异步 Promise


5️⃣ this 指向

CommonJS ES Module
顶层 this 指向 module.exports undefined

6️⃣ 循环依赖处理

CommonJS ES Module
行为 返回已执行部分 更安全(引用绑定)

👉 ESM 更不容易出问题


7️⃣ 语法差异

CommonJS ES Module
导入 require() import
导出 module.exports export

三、总结一句话(面试用)

👉 可以直接背这个:

CommonJS 是运行时同步加载,导出的是值的拷贝;

ES Module 是编译时静态加载,导出的是引用绑定,支持 Tree Shaking 和更好的优化。


四、结合你前端经验(加分理解)

你现在用的:

  • React / Vue / Vite / Webpack

其实底层:

  • 打包工具(如 Vite、Webpack)都会优先解析 ES Module

  • 因为:

    • 可以 Tree Shaking

    • 可以做代码分割(Code Splitting)


五、再补一个容易被问的点

Node.js 中如何区分 CJS 和 ESM?

复制代码
{
  "type": "module"
}
  • "module" → 使用 ES Module

  • "commonjs"(默认)→ 使用 CommonJS

相关推荐
一份执念9 分钟前
uni-app项目 (vue+vite + uni-UI)中引入umd格式JS文件,微信小程序中导入报错处理方案
前端·uni-app·echarts
To_OC13 分钟前
手写快排次次翻车?别死背快排模板了,这才是面试官想听的底层逻辑
javascript·算法·排序算法
ClouGence19 分钟前
2026 年自动化测试工具选型指南:8 款主流工具对比
前端·测试
lichenyang4531 小时前
为什么需要双线程通信、JavaScriptProxy 和 runJavaScript 分别干什么
前端
以和为贵1 小时前
前端也能搞懂 RAG:用 JS 手写一条最小检索增强链路
前端·人工智能·面试
风止何安啊1 小时前
网课倍速痛点解决:一套前端代码实现自由控速播放器
前端·javascript·node.js
牧艺1 小时前
用 Next.js + React Three Fiber 打造 3D 快递仓储可视化
前端·three.js
锋行天下2 小时前
如何用Vite实现Vue组件的按需打包和远程加载
前端·vue.js·前端框架
光影少年2 小时前
原生DOM操作在React 中的注意事项
前端·javascript·react.js
糖拌西瓜皮3 小时前
Node.js核心模块实战:文件、路径、HTTP与流处理
javascript·node.js