news 2026/6/22 17:44:56

【node.js】node.js 两种模块规范 CommonJS 和 ESM 如何选择?

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【node.js】node.js 两种模块规范 CommonJS 和 ESM 如何选择?

在 Node.js 的生态系统中,存在两种主要的模块规范:CommonJS (CJS)ES Modules (ESM)

理解它们的区别对于现代前端和后端开发至关重要,因为 Node.js 正在从 CJS 全面转向 ESM。


1. CommonJS (CJS)

CommonJS 是 Node.js 诞生的默认模块系统,已经存在了十多年。

  • 语法:使用require()导入模块,使用module.exports导出模块。
  • 加载方式同步加载。这意味着当你调用require()时,Node.js 会阻塞后续代码的执行,直到模块加载完成。
  • 适用场景:传统 Node.js 项目、服务器端脚本。

示例代码:

// 导出 (math.js)constadd=(a,b)=>a+b;module.exports={add};// 导入 (main.js)const{add}=require('./math.js');console.log(add(2,3));

2. ES Modules (ESM)

ESM 是 JavaScript 在 2015 年(ES6)确立的标准官方模块系统,旨在让浏览器和服务器通用一套标准。

  • 语法:使用import导入,使用export导出。
  • 加载方式异步加载。它分为三个阶段:构建(解析)、链接、求值。这种机制支持“静态分析”。
  • 适用场景:现代前端框架(Vue, React)、浏览器环境、现代 Node.js 项目。

示例代码:

// 导出 (math.js)exportconstadd=(a,b)=>a+b;// 导入 (main.js)import{add}from'./math.js';console.log(add(2,3));

3. 核心区别对比

特性CommonJS (CJS)ES Modules (ESM)
关键字require/module.exportsimport/export
加载时机运行时加载。编译时加载(静态分析)。
加载方式同步。异步。
顶层变量__dirname,__filename没有这些变量(需用import.meta.url模拟)。
this 指向指向当前模块。undefined
Tree Shaking不支持(因为是动态加载)。支持(可以删除未使用的代码)。
严格模式默认不开启。强制开启(“use strict”)。

4. 关键差异深度解析

(1) 静态分析 vs 动态加载
  • ESM 是静态的import语句必须放在文件顶部。因为它是静态的,打包工具(如 Webpack, Rollup, Vite)可以在代码运行前知道哪些代码被使用了,从而进行Tree Shaking(剔除无效代码),缩小包体积。
  • CJS 是动态的:你可以在if语句或函数内部写require()。这使得它很灵活,但也无法在编译阶段进行优化。
(2) 值的拷贝 vs 值的引用
  • CJS:导出的是值的拷贝。一旦输出一个值,模块内部的变化不会影响到已加载的值。
  • ESM:导出的是值的动态只读引用(Live Bindings)。如果模块内部修改了变量,外部引用的地方也会同步更新(但外部不能修改它)。
(3) 环境变量(__dirname)

在 CJS 中,你可以直接使用__dirname获取当前目录。在 ESM 中,这些全局变量不存在,你需要这样操作:

// ESM 获取 __dirname 的方法import{fileURLToPath}from'url';import{dirname}from'path';const__filename=fileURLToPath(import.meta.url);const__dirname=dirname(__filename);

5. 如何在 Node.js 中选择?

现在 Node.js 同时支持两者,判断规则如下:

  1. 文件名后缀
    • .cjs总是被视为 CommonJS。
    • .mjs总是被视为 ES Modules。
    • .js取决于最近的package.json
  2. package.json
    • 设置"type": "module",则.js文件被视为ESM
    • 设置"type": "commonjs"或不设置,则.js文件被视为CJS

6. 互操作性(能不能混用?)

  • ESM 导入 CJS:可以。import cjs from './file.cjs'通常能工作,但只能默认导入,不能通过解构导入(因为 CJS 是运行时生成的)。
  • CJS 导入 ESM不可以使用require()。因为 ESM 是异步的,而require是同步的。如果必须在 CJS 中用 ESM,只能使用异步的import()函数:
    // 在 CJS 文件中import('./esm-file.mjs').then(module=>{// 使用模块});

总结

  • 如果你正在开发新项目,建议优先使用ESM,因为它是未来的标准,且支持更好的性能优化。
  • 如果你在维护老旧项目或使用大量仅支持 CJS 的老旧库,则继续使用CommonJS
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/15 17:18:24

饮食健康管理|基于springboot + vue饮食健康管理系统(源码+数据库+文档)

饮食健康管理 目录 基于springboot vue饮食健康管理系统 一、前言 二、系统功能演示 三、技术选型 四、其他项目参考 五、代码参考 六、测试参考 七、最新计算机毕设选题推荐 八、源码获取: 基于springboot vue饮食健康管理系统 一、前言 博主介绍&…

作者头像 李华
网站建设 2026/6/21 21:20:04

Excalidraw周边产品创意设计展示

Excalidraw 周边产品创意设计的技术融合与实践探索 在远程协作成为常态的今天,团队沟通早已不再局限于文字和语音。一张随手画出的草图,往往比千言万语更能传达想法的本质。然而,传统的绘图工具要么过于正式、操作繁琐,要么缺乏灵…

作者头像 李华
网站建设 2026/6/16 20:35:46

Excalidraw时间线功能实战:项目进度可视化

Excalidraw时间线功能实战:项目进度可视化 在一次产品迭代会议中,团队正为即将到来的版本发布焦头烂额。项目经理打开Jira,展示了一长串任务列表;设计师却指着白板上的草图说:“但我们上周不是已经画好了流程吗&#x…

作者头像 李华
网站建设 2026/6/22 17:21:21

Excalidraw数据库选型建议(SQLite/PostgreSQL)

Excalidraw 数据库选型建议(SQLite vs PostgreSQL) 在现代团队协作中,可视化工具早已不再是“锦上添花”,而是项目推进的核心载体。Excalidraw 以其手绘风格和极简交互脱颖而出,成为架构设计、原型讨论甚至教学演示中…

作者头像 李华
网站建设 2026/6/22 14:30:36

ue 打包 实战笔记

打包linux:UE 编辑器菜单,编辑 → 项目设置左侧滚动到底平台 → Turnkey打包linux 平台:在 UE 编辑器里启用 Linux 平台打开 UE(你要打包的那个版本,比如 UE 5.1):菜单栏 → 编辑插件搜索&#…

作者头像 李华
网站建设 2026/6/15 4:21:24

Open-AutoGLM智能购票实战指南(从零搭建高并发抢票系统)

第一章:Open-AutoGLM智能购票系统概述Open-AutoGLM 是一款基于大语言模型与自动化流程控制的智能购票系统,专为应对高并发、强竞争的票务场景设计。系统融合自然语言理解、动态验证码识别、多平台协议适配与智能调度算法,实现从用户需求解析到…

作者头像 李华