ES6(ECMAScript 2015)引入的模块系统(Modules)是 JavaScript 历史上最重要的特性之一。它解决了长期存在的全局变量污染、依赖管理混乱等问题,为现代前端工程化奠定了基础。
本文将系统讲解 ES6 模块的导入/导出语法、运行机制、常见误区及最佳实践,助你写出清晰、高效、可维护的模块化代码。
一、为什么需要模块?
❌ 传统脚本的问题
<scriptsrc="utils.js"></script><scriptsrc="app.js"></script><!-- 顺序敏感!utils 必须在 app 前 --><!-- 所有变量挂载到全局 window,易冲突 -->✅ ES6 模块的优势
- 作用域隔离:模块内变量不会污染全局;
- 显式依赖:通过
import/export声明依赖关系; - 静态分析:编译时确定依赖(支持 Tree-shaking);
- 异步加载:天然支持按需加载(配合动态 import)。
🔑核心原则:每个文件就是一个模块(需在
<script type="module">或构建工具中使用)。
二、导出(Export)详解
1. 命名导出(Named Export)
可导出多个值(变量、函数、类等),名称必须匹配。
// math.jsexportconstPI=3.14159;exportfunctionadd(a,b){returna+b;}exportclassCalculator{...}// 或统一导出constsubtract=(a,b)=>a-b;functionmultiply(a,b){returna*b;}export{subtract,multiply};✅特点:
- 可多次使用
export; - 导入时必须用相同名称(或重命名);
- 支持重命名导出:
export { foo as bar }。
2. 默认导出(Default Export)
每个模块只能有一个默认导出,常用于导出主功能(如组件、类、函数)。
// Button.vue (伪代码)exportdefaultfunctionButton(props){return`<button>${props.text}</button>`;}// 或constMyComponent={...};exportdefaultMyComponent;✅特点:
- 导入时可自定义名称;
- 语法简洁:
export default expression。
3. 混合导出(不推荐)
// utils.jsexportdefaultfunctionmain(){...}exportconsthelper=()=>{...};⚠️ 虽合法,但易造成 API 混乱,建议统一使用命名导出或仅用默认导出。
三、导入(Import)详解
1. 导入命名导出
// 方式1:逐个导入import{PI,add}from'./math.js';// 方式2:重命名(避免冲突)import{addassum,subtractasminus}from'./math.js';// 方式3:导入所有为命名空间对象import*asMathUtilsfrom'./math.js';MathUtils.add(1,2);// 调用方式2. 导入默认导出
// 名称可自定义importButtonfrom'./Button.js';importMyCompfrom'./MyComponent.js';3. 同时导入默认和命名导出
// 默认导出在前,命名导出在后importButton,{variant,size}from'./Button.js';4. 仅执行模块(无绑定导入)
// 仅运行模块代码(如初始化、副作用)import'./polyfills.js';import'./analytics.js';// 初始化埋点四、动态导入(Dynamic Import)
用于按需加载、条件加载或懒加载,返回 Promise。
// 动态导入语法constmodule=awaitimport('./math.js');console.log(module.add(1,2));// 条件加载if(featureFlag){const{newFeature}=awaitimport('./new-feature.js');newFeature();}// React 懒加载组件constLazyComponent=React.lazy(()=>import('./LazyComponent'));✅优势:
- 减少首屏 bundle 体积;
- 实现代码分割(Code Splitting);
- 兼容性处理(如 polyfill 按需加载)。
五、关键特性与注意事项
1.静态结构(Static Structure)
import/export必须在顶层作用域,不能在条件语句或函数内:// ❌ 错误!if(true){import{foo}from'./foo.js';// SyntaxError}- 原因:便于静态分析(Tree-shaking、依赖图构建)。
✅ 解决方案:使用动态导入实现条件加载。
2.实时绑定(Live Binding)
命名导出是只读的实时引用,不是值拷贝:
// counter.jsexportletcount=0;exportfunctionincrement(){count++;}// main.jsimport{count,increment}from'./counter.js';console.log(count);// 0increment();console.log(count);// 1 —— 自动更新!⚠️ 默认导出是值拷贝(除非导出的是引用类型)。
3.文件扩展名要求
- 在浏览器原生使用时,必须包含
.js扩展名:// ✅ 正确import{foo}from'./utils.js';// ❌ 错误(浏览器会 404)import{foo}from'./utils'; - 构建工具(如 Vite、Webpack)可省略扩展名(自动解析)。
4.MIME 类型要求(浏览器)
HTML 中必须声明type="module":
<scripttype="module"src="./main.js"></script>否则浏览器会当作传统脚本执行,导致
import报错。
六、最佳实践
✅ 1.优先使用命名导出
- 明确 API 边界,利于 Tree-shaking;
- 避免“猜名字”(默认导出名称随意);
- 支持 IDE 自动补全和重构。
// 推荐export{Button,IconButton,LinkButton};// 而非exportdefault{Button,IconButton,LinkButton};✅ 2.避免混合导出
- 一个模块要么全用命名导出,要么只用默认导出;
- 例外:库的主入口可默认导出主类,同时命名导出工具函数(如 Lodash)。
✅ 3.使用as重命名解决冲突
import{debounceas_debounce}from'lodash';import{debounceasmyDebounce}from'./utils';✅ 4.目录结构与 barrel 文件
用index.js聚合子模块,简化导入路径:
// components/index.jsexport{defaultasButton}from'./Button/Button.js';export{defaultasModal}from'./Modal/Modal.js';// 使用import{Button,Modal}from'./components';✅ 5.动态导入用于性能优化
- 路由级代码分割(React Router、Vue Router);
- 大型工具库按需加载(如
moment.js→dayjs); - A/B 测试、实验性功能。
七、常见误区
❌ 误区1:import是解构赋值
// 错误理解import{foo}from'./mod';// 不是解构!// 正确:这是静态导入声明,绑定到模块的导出❌ 误区2:默认导出更“高级”
- 默认导出只是语法糖,没有性能或功能优势;
- 过度使用会导致 API 不清晰(如
import Whatever from 'lib')。
❌ 误区3:可以修改导入的绑定
import{PI}from'./math.js';PI=3;// ❌ TypeError: Assignment to constant variable.命名导入是只读绑定!
八、总结:ES6 模块黄金法则
| 场景 | 推荐做法 |
|---|---|
| 导出多个工具函数/类 | 命名导出 |
| 导出单一主组件/类 | 默认导出 |
| 需要 Tree-shaking | 避免* as,使用具体命名导入 |
| 条件加载/懒加载 | 动态import() |
| 统一 API 入口 | Barrel 文件(index.js) |
| 浏览器原生使用 | 带.js扩展名 +type="module" |
💡记住:
“显式优于隐式,静态优于动态,命名优于默认”—— 这是写出高质量模块化代码的核心思想。
掌握 ES6 模块,是迈向现代 JavaScript 开发的第一步。合理使用import/export,让你的代码更清晰、更安全、更高效!