ES模块的export必须在顶层声明,不可在条件语句中;import路径需带扩展名或符合解析规则;default与named export需分开导入;动态import()返回Promise,须异步处理。
ES 模块的 export 必须是顶层声明,不能包裹在 if、for 或函数里。否则会报错 Unexpected token 'export' 或 Cannot use import statement outside a module(后者常因运行环境未启用模块模式导致)。
export const API_URL = 'https://api.example.com';
export function fetchData() { return fetch(API_URL); }
export default class Logger { log(msg) { console.log(msg); } }
if (process.env.NODE_ENV === 'dev') {
export const DEBUG = true; // SyntaxError
}
const features = {
debug: process.env.NODE_ENV === 'dev',
analytics: true
};
export { features };
在浏览器原生 ESM 中,import 的路径必须是相对路径(./ 或 ../)、绝对路径(/)或完整 URL;且不能省略扩展名(如 .js),除非服务器配置了 MIME 类型和重定向规则。
import { foo } from './utils.js';
import React from 'https://cdn.skypack.dev/react@18';
import { foo } from './utils'; // 报 404:找不到 ./utils
import { bar } from 'lodash'; // 报错:不是有效 URL,也不在 import map 中
从 v14.13 起支持 .mjs 或 type: "module" 的 package.json;若用 .js 文件,必须显式声明 "type": "module",否则 import 会报错。
一个模块可以同时有 export default 和多个 export(具名),但导入时不能把它们混在同一个 import 语句里,也不能用相同名字重复声明。
export default function main() {}
export const VERSION = '1.0';
export function helper() {}
import main, { VERSION, helper } from './app.js'; // ✅ 正确
import { default as main, VERSION } from './app.js'; // ✅ 也可,但不常用
import main, { VERSION } from './app.js'; // ✅ 同上
import { main, VERSION } from './app.js'; // ❌ main 不是具名导出,会是 undefined
default 的具名导出,只是语法糖;所以以下两种等价:export default function foo() {}
// 等价于
function foo() {}
export { foo as default };
import() 返回 Promise,适用于按需加载、条件加载或拆包场景,但它不是同步 import 的替代品。直接赋值或想立刻调用会出问题。
const module = import('./math.js'); // module 是 Promise,不是模块对象
console.log(module.add); // undefined
async function loadMath() {
const math = await import('./math.js');
return math.add(2, 3);
}
// 或
import('./math.js').then(m => m.add(2, 3));
import() 在 Node.js v12+ 和现代浏览器中支持良好,但 IE 完全不支
持,旧版 Safari 需要检查。模块路径仍需遵守前述规则(如带 .js 扩展名),且不能传变量拼接的字符串(如 import(`./${name}.js`))——除非构建工具(如 Webpack/Vite)做了特殊支持,原生 ESM 不允许表达式。