概述
历史上,JavaScript 一直没有模块(module)体系,无法将一个大程序拆分成互相依赖的小文件,再用简单的方法拼装起来。其他语言都有这项功能,比如 Ruby 的require、Python 的import,甚至就连 CSS 都有@import,但是 JavaScript 任何这方面的支持都没有,这对开发大型的、复杂的项目形成了巨大障碍。
在 ES6 之前,社区制定了一些模块加载方案,最主要的有 CommonJS 和 AMD 两种。前者用于服务器,后者用于浏览器。ES6 在语言标准的层面上,实现了模块功能,而且实现得相当简单,完全可以取代 CommonJS 和 AMD 规范,成为浏览器和服务器通用的模块解决方案。
ES6 模块的设计思想是尽量的静态化,使得编译时就能确定模块的依赖关系,以及输入和输出的变量。CommonJS 和 AMD 模块,都只能在运行时确定这些东西。比如,CommonJS 模块就是对象,输入时必须查找对象属性。
模块功能主要由两个命令构成:export和import。export命令用于规定模块的对外接口(即导出模块),import命令用于输入模块(即导入模块)。
其中使用import命令的时候,用户需要知道所要加载的变量名或函数名,否则无法加载,所以ES6为用户提供了更方便的方法export default
用法
其中 export default、export、import 属于ES6产物,module.exports和require属于node.js产物。
node中导入模块:var 名称 = require('模块标识符')
node中向外暴露成员的形式:module.exports = {}
export
用法1
```
export var firstName = 'Michael';
export var lastName = 'Jackson';
export var year = 1958;
```
上面代码ES6 将其视为一个模块,里面用export命令对外部输出了三个变量。
用法2
```
var firstName = 'Michael';
var lastName = 'Jackson';
var year = 1958;
```
export { firstName, lastName, year };
上面代码在export命令后面,使用大括号指定所要输出的一组变量。它与前一种写法(直接放置在var语句前)是等价的,但是应该优先考虑使用这种写法。因为这样就可以在脚本尾部,一眼看清楚输出了哪些变量。
用法3
```
export function multiply(x, y) {
return x * y;
}
```
//export命令除了输出变量,还可以输出函数或类(class)
通常情况下,export输出的变量就是本来的名字,但是可以使用as关键字重命名。
```
function v1() { ... }
function v2() { ... }
export {
v1 as streamV1,
v2 as streamV2,
v2 as streamLatestVersion
};
```
//上面代码使用as关键字,重命名了函数v1和v2的对外接口。重命名后,v2可以用不同的名字输出两次。
//as的作用 => 将 xxx 重新命名为 xxx
需要特别注意的是,export命令规定的是对外的接口,必须与模块内部的变量建立一一对应关系。
```
// 报错
export 1;
// 报错
var m = 1;
export m;
```
//上面两种写法都会报错,因为没有提供对外的接口。第一种写法直接输出 1,第二种写法通过变量m,还是直接输出 1
也就是说用export 导出模块的时候,如果变量或函数提前定义好了导出是必须加 {}, 如果没有提前定义导出是必须以命名(定义的)的方式
```
// 报错
export 1;
// 报错
var m = 1;
export m;
// 报错
function f() {}
export f;
// 正确
export var m = 1;
// 正确
var m = 1;
export {m};
// 正确
var n = 1;
export {n as m};
// 正确
export function f() {};
// 正确
function f() {}
export {f};
import
```
注意2点
1、由于import是静态执行,所以不能使用表达式和变量,这些只有在运行时才能得到结果的语法结构。
```
// 报错
import { 'f' + 'oo' } from 'my_module';
// 报错
let module = 'my_module';
import { foo } from module;
// 报错
if (x === 1) {
import { foo } from 'module1';
} else {
import { foo } from 'module2';
}
```
2、模块的整体加载 (除了指定加载某个输出值,还可以使用整体加载,即用星号(*)指定一个对象,所有输出值都加载在这个对象上面)
```
export function area(radius) {
return Math.PI * radius * radius;
}
export function circumference(radius) {
return 2 * Math.PI * radius;
}
```
import 导入上面代码所有值方式
```
import * as circle from './circle';
console.log('圆面积:' + circle.area(4));
console.log('圆周长:' + circle.circumference(14));
export default
```
export default命令,为模块指定默认输出。export default如果导出变量时必须时匿名,如果导出函数时可以时命名也可以匿名
导出变量
```
// 正确
var a = 1;
export default a;
//或者
export default 1; //这样做没有实际意思
// 错误
export default var a = 1;
```
导出函数
```
export default function () {
console.log('foo');
}
//或者
export default function foo() {
console.log('foo');
}
// 或者
function foo() {
console.log('foo');
}
export default foo;
```
export default命令用于指定模块的默认输出。显然,一个模块只能有一个默认输出,因此export default命令只能使用一次。所以,import命令后面才不用加大括号,因为只可能唯一对应export default命令。
本质上,export default就是输出一个叫做default的变量或方法,然后系统允许你为它取任意名字。所以,下面的写法是有效的。
所以通过以上用法可以总结export default和export区别
1、export default 向外暴露的成员,可以使用任意变量来接收
2、在一个模块中,export default 只允许向外暴露一次
3、在一个模块中,可以同时使用export default 和export 向外暴露成员
4、使用export向外暴露的成员,只能使用{ }的形式来接收,这种形式,叫做【按需导出】
5、export可以向外暴露多个成员,同时,如果某些成员,在import导入时,不需要,可以不在{ }中定义
6、使用export导出的成员,必须严格按照导出时候的名称,来使用{ }按需接收
7、使用export导出的成员,如果想换个变量名称接收,可以使用as来起别名
import语句和import()属于ES6语法,而require()属于node中的。其中import()是ES2020引入函数,支持动态加载模块。
import命令能够接受什么参数,import()函数就能接受什么参数,两者区别主要是后者为动态加载。
import是静态加载资源,编译时放到代码块最顶层。
import()函数则是动态按需加载返回Promise 对象。import()函数可以用在任何地方,不仅仅是模块,非模块的脚本也可以使用。
import()函数它是运行时执行,也就是说,什么时候运行到这一句,就会加载指定的模块。
import()函数与所加载的模块没有静态连接关系,这点也是与import语句不相同。import()类似于 Node 的require方法,区别主要是前者是异步加载,后者是同步加载。
| | 参数 | 是否动态加载 | 与所加载的模块是否有静态连接关系 |
| --- | --- |--- |--- |
| import语句 | 引入的地址路径 |否 |是 |
| import() | 引入的地址路径 | 是(异步加载) | 否 |
| require() | 引入的地址路径 | 是(同步加载) | 否 |
- 介绍
- 快速了解
- 项目简介
- 主要特性
- 技术选型
- 内置功能
- 更新日志
- 环境部署
- 准备工作
- 运行系统
- 部署系统
- 环境变量
- nginx配置
- 项目介绍
- 文件结构
- 核心技术
- 技术文档
- 多语言环境配置
- 如何在vue项目中整合tinymce富文本编辑器
- vuedraggable在项目中的应用
- viewerjs在项目中的应用
- 用echart在vue项目中实现数据可视化
- 用webpack打包vue项目时如何实现性能调优
- CSS高度塌陷原理及解决方法
- CSS的几种定位机制
- 话说BFC
- export、export default和module.exports的用法及区别
- proto 和 prototype 深度剖析
- 服务端渲染的探索与实践
- 浏览器背后的运行机制
- 组件文档
- 基础组件
- 扩展按钮
- 扩展表格
- 定制面包屑
- 超级图片上传
- 地图定位
- 换肤调色板
- 富文本编辑器
- 视频上传
- 图片裁剪
- 表格树状列组件
- 自定义显示列
- 业务组件
- 更换头像
- 图片排序
- 地域选择
- 选择文章
- 文章分类选择
- 表单选择
- 商品选择
- 常见问题
- 捐赠支持
- 演示截图
- 功能列表