1. 概述
在ES6之前,模块加载的方案主要有CommonJS和AMD。前者用于服务器,后者用于浏览器。ES6在语言规格层面上实现了模块化,直接用import和export在浏览器中导入和导出各个模块,一个js文件代表一个js模块,相当简洁,可以完全取代现有的CommonJS和AMD规范。
2.1 export命令
模块功能主要由两个命令构成:export和import。export命令用于规定模块的对外接口,import命令用于输入其他模块提供的功能。
一个模块就是一个独立的文件。该文件内部的所有变量,外部无法获取。如果你希望外部能够读取模块内部的某个变量,就必须使用export关键字输出该变量。如下JS文件,里面使用export命令输出变量。
|
|
上面代码是profile.js文件,保存了年份信息。ES6将其视为一个模块,里面用export命令对外部输出了该变量。
- export的写法,除了像上面这样,还有另外一种。
|
|
上面代码在export命令后面,使用大括号指定所要输出的变量。它与前一种写法(直接放置在var语句前)是等价的,但是应该优先考虑使用这种写法。因为这样就可以在脚本尾部,一眼看清楚输出了哪些变量。
- export命令除了输出变量,还可以输出函数或类(class)。
|
|
- 通常情况下,export输出的变量就是本来的名字,但是可以使用as关键字重命名。并且重命名后,可以用不同的名字输出两次。
|
|
- 另外,export命令规定的是对外的接口,必须与模块内部的变量建立一一对应关系。
|
|
上面两种写法都会报错,因为没有提供对外的接口。第一种写法直接输出 1,第二种写法通过变量m,还是直接输出1。1只是一个值,不是接口。正确的写法是下面这样。
|
|
同样的,function和class的输出,也必须遵守这样的写法。
- export语句输出的接口,与其对应的值是动态绑定关系,即通过该接口,可以取到模块内部实时的值。
|
|
上面代码输出变量foo,值为bar,500 毫秒之后变成baz。
最后,export命令可以出现在模块的任何位置,只要处于模块顶层就可以。如果处于块级作用域内,就会报错,import命令也是如此。
|
|
上面代码中,export语句放在函数之中,结果报错。因为处于条件代码块之中,就没法做静态优化了,违背了 ES6 模块的设计初衷。
2.2 import命令
使用export命令定义了模块的对外接口以后,其他JS文件就可以通过import命令加载这个模块。
|
|
上面代码的import命令,用于加载profile.js文件,并从中输入变量。import命令接受一对大括号,里面指定要从其他模块导入的变量名。大括号里面的变量名,必须与被导入模块(profile.js)对外接口的名称相同。
- 如果想为输入的变量重新取一个名字,import命令要使用as关键字,将输入的变量重命名。
|
|
import命令输入的变量都是只读的,因为它的本质是输入接口。也就是说,不允许在加载模块的脚本里面,改写接口。
|
|
上面代码中,脚本加载了变量a,对其重新赋值就会报错,因为a是一个只读的接口。但是,如果a是一个对象,改写a的属性是允许的。
|
|
上面代码中,a的属性可以成功改写,并且其他模块也可以读到改写后的值。
另外,import后面的from指定模块文件的位置,可以是相对路径,也可以是绝对路径,.js后缀可以省略。import命令具有提升效果,会提升到整个模块的头部,首先执行。
|
|
上面的代码不会报错,因为import的执行早于foo的调用。这种行为的本质是,import命令是编译阶段执行的,在代码运行之前。
由于import是静态执行,所以不能使用表达式和变量,这些只有在运行时才能得到结果的语法结构。
|
|
上面三种写法都会报错,因为它们用到了表达式、变量和if结构。
最后,import语句会执行所加载的模块,因此可以有下面的写法。
|
|
如果多次重复执行同一句import语句,那么只会执行一次,而不会执行多次。
|
|
上面代码加载了两次lodash,但是只会执行一次。
|
|
上面代码中,虽然foo和bar在两个语句中加载,但是它们对应的是同一个my_module实例。
2.3 模块的整体加载
除了指定加载某个输出值,还可以使用整体加载,即用星号(*)指定一个对象,所有输出值都加载在这个对象上面。
如下circle.js文件,它输出两个方法area和circumference。
|
|
下面,加载这个模块。
|
|
上面写法是逐一指定要加载的方法,整体加载的写法如下。
|
|
注意,模块整体加载所在的那个对象(上例是circle),应该是可以静态分析的,所以不允许运行时改变。下面的写法都是不允许的。
|
|
2.4 export default命令
使用export default命令,可以为模块指定默认输出。
|
|
上面代码是一个模块文件export-default.js,它的默认输出是一个函数。
其他模块加载该模块时,import命令可以为该匿名函数指定任意名字。
|
|
上面代码的import命令,可以用任意名称指向export-default.js输出的方法,这时就不需要知道原模块输出的函数名。这时import命令后面,不使用大括号。
export default命令用在非匿名函数前,也是可以的。
|
|
上面代码中,foo函数的函数名foo,在模块外部是无效的。加载的时候,视同匿名函数加载。
下面比较一下默认输出和正常输出。
|
|
上面代码的两组写法,第一组是使用export default时,对应的import语句不需要使用大括号;第二组是不使用export default时,对应的import语句需要使用大括号。
- export default命令用于指定模块的默认输出。显然,一个模块只能有一个默认输出,因此export default命令只能使用一次。所以,import命令后面才不用加大括号,因为只可能唯一对应export default命令。本质上,export default就是输出一个叫做default的变量或方法,然后系统允许你为它取任意名字。所以,下面的写法是有效的。
|
|
- 因为export default命令其实只是输出一个叫做default的变量,所以它后面不能跟变量声明语句。
|
|
上面代码中,export default a的含义是将变量a的值赋给变量default。所以,最后一种写法会报错。
- 同样地,因为export default命令的本质是将后面的值,赋给default变量,所以可以直接将一个值写在export default之后。
|
|
上面代码中,后一句报错是因为没有指定对外的接口,而前一句指定外对接口为default。
- 如果想在一条import语句中,同时输入默认方法和其他接口,可以写成下面这样。
|
|
对应上面代码的export语句如下。
|
|
上面代码的最后一行的意思是,暴露出forEach接口,默认指向each接口,即forEach和each指向同一个方法。
- export default也可以用来输出类。
|
|
2.5 export 与 import 的复合写法
如果在一个模块之中,先输入后输出同一个模块,import语句可以与export语句写在一起。
|
|
上面代码中,export和import语句可以结合在一起,写成一行。但需要注意的是,写成一行以后,foo和bar实际上并没有被导入当前模块,只是相当于对外转发了这两个接口,当前模块不能直接使用foo和bar。
- 模块的接口改名和整体输出,也可以采用这种写法。
|
|
- 具名接口改为默认接口的写法如下。
|
|
- 同样地,默认接口也可以改名为具名接口。
|
|
下面三种import语句,没有对应的复合写法。
|
|
3.1 模块的继承
模块之间可以继承。
假设有一个circleplus模块,继承了circle模块。
|
|
上面代码中的export ,表示再输出circle模块的所有属性和方法。注意,export 命令会忽略circle模块的default方法。然后,上面代码又输出了自定义的e变量和默认方法。
- 也可以将circle的属性或方法,改名后再输出。
|
|
加载上面模块的写法如下。
|
|
上面代码中的import exp表示,将circleplus模块的默认方法加载为exp方法。
3.2 跨模块常量
我们知道,const声明的常量只在当前代码块有效。如果想设置跨模块的常量(即跨多个文件),或者说一个值要被多个模块共享,可以采用下面的写法。
|
|
如果要使用的常量非常多,可以建一个专门的constants目录,将各种常量写在不同的文件里面,保存在该目录下。
|
|
然后,将这些文件输出的常量,合并在index.js里面。
|
|
使用的时候,直接加载index.js就可以了。
|
|
4.1 import()
前面说过,import命令会被 JavaScript 引擎静态分析,先于模块内的其他模块执行。所以,下面的代码会报错。
|
|
上面代码中,引擎处理import语句是在编译时,这时不会去分析或执行if语句,所以import语句放在if代码块之中毫无意义,因此会报句法错误,而不是执行时错误。
|
|
上面的语句就是动态加载,require到底加载哪一个模块,只有运行时才知道。import语句做不到这一点。
因此,有一个提案,建议引入import()函数,完成动态加载。
|
|
上面代码中,import函数的参数specifier,指定所要加载的模块的位置。import命令能够接受什么参数,import()函数就能接受什么参数,两者区别主要是后者为动态加载。
import()返回一个 Promise 对象。下面是一个例子。
|
|
- import()函数可以用在任何地方,不仅仅是模块,非模块的脚本也可以使用。它是运行时执行,也就是说,什么时候运行到这一句,也会加载指定的模块。
4.2 适用场合
(1)按需加载。import()可以在需要的时候,再加载某个模块。
|
|
上面代码中,import()方法放在click事件的监听函数之中,只有用户点击了按钮,才会加载这个模块。
(2)条件加载。import()可以放在if代码块,根据不同的情况,加载不同的模块。
|
|
上面代码中,如果满足条件,就加载模块 A,否则加载模块 B。
(3)动态的模块路径。import()允许模块路径动态生成。
|
|
上面代码中,根据函数f的返回结果,加载不同的模块。
4.3 注意点
- import()加载模块成功以后,这个模块会作为一个对象,当作then方法的参数。因此,可以使用对象解构赋值的语法,获取输出接口。
|
|
上面代码中,export1和export2都是myModule.js的输出接口,可以解构获得。
- 模块有default输出接口,可以用参数直接获得。
|
|
- 如果想同时加载多个模块,可以采用下面的写法。
|
|
- import()也可以用在 async 函数之中。
|
|
未完待续……
参考书籍:《ES6标准入门(第3版)》