# 为什么会存在 CJS 和 ESM 混合开发?
npm 模块有的使用 CJS 有的使用 ESM,导致 CJS 和 ESM 混合开发成为 Node 项目必须考虑的问题
# 场景 1:CommonJS 单独使用
CJS 文件定义方法:
- 默认 .js 后缀的文件为 CJS。
- .cjs 后缀的文件强制为 CJS。
- package.json 中 type="commonjs" 时,.js 文件会被认为是 CJS。
踩坑:module.exports 和 exports.fn 混用问题,注意 exports 是 module.exports 的引用,如果引用变了导出就会失效
# 场景 2:ESM 单独使用
ESM 文件定义方法:
- package.json 中 type="module" 时,.js 后缀文件会被识别为 ESM。
- .mjs 后缀的文件会被强制识别为 ESM,需要 Node.js 14 及以上版本才支持,如果是 14 以下,命令运行时需要加 --experimental-modules 参数
踩坑:**filename, **dirname 无法使用,解决方案 (opens new window)
// In ".mjs" script or a script under "type": "module" package
import { dirname, filename } from 'dirname-filename-esm'
const __dirname = dirname(import.meta)
const __filename = filename(import.meta)
// Use __dirname and __filename like under commonjs module
1
2
3
4
5
6
7
2
3
4
5
6
7
# 场景 3:CommonJS 和 ESM 混用
原则:
- 单个模块必须指定 CommonJS 和 ESM,不可混用,如果混用,必须使用 webpack 或者 babel 进行解决
- package.json 的 type 必须指定一种模块,如果不指定默认为 commonjs,但越来越多的模块采用 module(ESM)
# 场景 3.1:CommonJS 兼容 ESM
import('esm').then(({ a, b }) => {
a()
b()
})
1
2
3
4
2
3
4
# 场景 3.2:ESM 兼容 CommonJS
import { a, b } from 'cjs'
a()
b()
1
2
3
4
2
3
4
# 常见错误
# 错误 1:type 和语法不匹配
常见错误:
node packages/esm_entry/
(node:16435) Warning: To load an ES module, set "type": "module" in the package.json or use the .mjs extension.
(Use `node --trace-warnings ...` to show where the warning was created)
/Users/sam/Desktop/fe_arch_source_code/imooc-test/module/packages/esm_entry/lib/index.js:1
import cjs from 'cjs';
^^^^^^
SyntaxError: Cannot use import statement outside a module
1
2
3
4
5
6
7
8
2
3
4
5
6
7
8
错误原因:
- 未指定 package.json 的 type,默认采用 commonjs
- js 源码中使用了 ESM 语法,如 import 或 export 等
解决方案:
- 修改 js 为 mjs(不建议使用)
- 修改 type 为 module
# 错误 2:require 无法加载 ESM 模块
使用 require 加载 ESM 模块会出现报错:
internal/modules/cjs/loader.js:1102
throw new ERR_REQUIRE_ESM(filename, parentPath, packageJsonPath);
^
Error [ERR_REQUIRE_ESM]: Must use import to load ES Module: /Users/sam/Desktop/fe_arch_source_code/imooc-test/module/packages/esm/lib/index.js
require() of ES modules is not supported.
1
2
3
4
5
6
2
3
4
5
6
解决方案:使用 import 方法加载
# 错误 3:ESM 模块加载其他模块问题
使用以下代码:
import a from './a'
1
会出现报错:
internal/process/esm_loader.js:74
internalBinding('errors').triggerUncaughtException(
^
Error [ERR_MODULE_NOT_FOUND]: Cannot find module '/Users/sam/Desktop/fe_arch_source_code/imooc-test/module/packages/esm/lib/a' imported from /Users/sam/Desktop/fe_arch_source_code/imooc-test/module/packages/esm/lib/index.js
1
2
3
4
5
2
3
4
5
解决方案:
import a from './a.js'
1
作者:chenjie
链接:https://webchenjie.cn
来源:ChenJieBlog