ES6 模块的设计思想是尽量的静态化,使得编译时就能确定模块的依赖关系,以及输入和输出的变量。
CommonJS 和 AMD 模块,都只能在运行时确定这些东西。比如,CommonJS 模块就是对象,输入时必须查找对象属性。

CommonJS模块

let { stat, exists, readFile } = require('fs');  
上面代码 等同于 
let _fs = require('fs'); 
let stat = _fs.stat; 
let exists = _fs.exists; 
let readfile = _fs.readfile;

上面代码的实质是整体加载fs模块(即加载fs的所有方法),生成一个对象(_fs),然后再从这个对象上面读取 3 个方法。这种加载称为“运行时加载”,
因为只有运行时才能得到这个对象,导致完全没办法在编译时做“静态优化”。

ES6模块

import { stat, exists, readFile } from 'fs';

上面代码的实质是从fs模块加载 3 个方法,其他方法不加载。这种加载称为“编译时加载”或者静态加载,即 ES6 可以在编译时就完成模块加载,
效率要比 CommonJS 模块的加载方式高。当然,这也导致了没法引用 ES6 模块本身,因为它不是对象。

CommonJS 加载的是一个对象(即module.exports属性),该对象只有在脚本运行完才会生成。
而 ES6 模块不是对象,它的对外接口只是一种静态定义,在代码静态解析阶段就会生成。

差异:
1. CommonJS 模块输出的是值的拷贝,也就是说,一旦输出一个值,模块内部的变化就影响不到这个值;ES6 模块的运行机制与 CommonJS 不一样。
 
2. JS 引擎对脚本静态分析的时候,遇到模块加载命令import,就会生成一个只读引用。
等到脚本真正执行时,再根据这个只读引用,到被加载的那个模块里面去取值。
原始值变了,import加载的值也会跟着变首先,就是this关键字。

3. ES6 模块之中,顶层的this指向undefined;
4. CommonJS 模块的顶层this指向当前模块,这是两者的一个重大差异。

es6 模块 导入导出

1. 第一种方式
// test.js

export  function fc(){
    console.log('fc');
}
export  function fd(){
    console.log('fd');
}

import {fa,fb} from './modules/test.js';

// 引用模块 也可以用 as 重新命名;
import {fa as myname,fb} from './modules/test.js';

// 也可以用*号作为整体加载;
import * as mymodule from './modules/test3.js';


2. 第二种
// test2.js

function t3(){
    console.log('t3');
}
function t4(){
    console.log('t4');
}
let obj = { t2,t3,t4 };

// 一个模块中 export default 只能用一次; 也可以用于具名函数;
export default obj

// 自己定义一个名字来接收 如果想这样接收,只能通过 export default 来暴露模块;
import mod from './modules/test2.js';