es6模块

模块化主要是帮助我们更好的组织代码, 模块允许我们将相关的变量和函数放在一个模块中。 在ES6模块化之前,JS语言并没有模块的概念,只有函数作用域和全局作用域非常容易发生命名冲突。 之前的RequireJS, SeaJS, AMD, UMD,CMD在一定层面上都是为了解决JS模块化的问题。ES6模块取其精华:

  • 它提供了简洁的语法
  • 以及异步的, 可配置的模块加载

什么是模块

模块是自动运行在严格模式下并且没有办法退出运行的JavaScript代码

  1. 在模块的顶部this的值是undefined
    2.其模块不支持html风格的代码注释
  2. 除非用default关键字,否则不能用这个语法导出匿名函数或类

任何未限制导出的变量、函数或类都是模块私有的,无法从模块外部访问

为什么要使用模块

目前最普遍的JS运行平台便是浏览器,在浏览器中,所有的代码都运行在同一个全局上下文中, 这使得你即使更改应用中的很小一部分, 你也要担心可能会产生的命名冲突。

传统的JS应用被分离在多个文件夹中,并且在构建的时候连接在一起,这稍显笨重。所以人们开始将每个文件内的代码都包在一个自执行函数中: (function(){ ... })(); 。 这种方法创建了一个本地作用域,于是最初的模块化的概念产生了, 之后的CommonJS和AMD系统中所称的模块, 也是由此实现的。

创建模块

一个JS模块就是一个对其他模块暴露一些内部的属性、方法的文件。 这里仅讨论浏览器中的ES2016模块系统。

每个模块都有自己的上下文

和传统的JS不同,在使用模块时,你不必担心污染全局作用域。恰恰相反,你需要把所有你需要用到的东西从其他模块中导入进来,这样会使得模块之间的依赖关系更为清晰

导入导出

可以使用ES6的新关键字 importexports 来导入或导出模块中的对象。 模块可以导入和导出各种类型的变量,如函数,对象,字符串,数字,布尔值等。

默认导出

每一个模块都支持导出一个不具名的变量,这为默认导出:

// helloWord.js
export default function(){
    console.log('111');
}

// main.js
import hello from './helloWord';
import anotherHello from './helloWord';

hello(); // 111
anotherHello(); //111
console.log(hello === anotherHello); //true

等价的CommonJS语法:

// helloWord.js
module.exports = function(){
    console.log('111');
}

//main.js
var hello = require('./helloWord');
var anotherHello = require('./helloWord');

hello(); // 111
anotherHello(); //111
console.log(hello === anotherHello); //true

任何JS值都是可以被默认导出的:

// helloWord.js
export default 3.14
//export default { foo: 'bar' };
//export default 'hello word';

// main.js
import hello from './helloWord';

console.log(hello); // 3.14

这里如果把注释放开,同样输出3.14, 当有多条export default语句,只会输出第一条export default的值

// helloWord.js
export default 3.14
export default { foo: 'bar' };
export default 'hello word';

// main.js
import hello from './helloWord';

console.log(hello); // 3.14
具名导入

除了默认导出外, ES6的模块系统还支持导出任意数量个具名的变量:

const PI = 3.14
const value = 42;
export function hello(){
    console.log('2111');
}
export {PI,value}

// 等同于CommonJS语法

// var PI = 3.14;
// var value = 42;
// module.exports.hello = function(){
//     console.log('2111');
// }
// module.exports.PI = PI;
// module.exports.value = value;

导入的时候:

import {PI,value,hello} from './helloWord';

console.log(PI,value,hello());
image.png

导入的时候可以使用as关键字来重命名导入的变量:

import {PI as PI2,value as val,hello as helloWord} from './helloWord';

console.log(PI2,val,helloWord());

结果是一样的

导入所有

最简单的,在一条命令中导入一个模块中所有变量的方法, 是使用*标记。 这样一来,被导入模块中所有导出的变量都会变成它的属性, 默认导出的变量则会被置于default属性中。

// helloWord.js
const PI = 3.14
const value = 42;
export function hello(){
    console.log('2111');
}
export {PI,value}

// main.js
import * as Hello from './helloWord';

console.log(Hello);
image.png

注意一点
import * as foo fromimport foo from的区别。 后者仅仅会导入默认导出的变量,而前者则会在一个对象中导入所有,如:

// helloWord.js
const PI = 3.14
const value = 42;
const foo = {
    'a':'123'
}
export function hello(){
    console.log('2111');
}
export {PI,value,foo}

// main.js
import * as foo from './helloWord';

console.log(foo);
console.log(foo.foo);
image.png

对比四种模块价值规范

  1. CommonJS
  2. AMD
  3. CMD
  4. ES6模块
1. CommonJS

commonJS是服务器端的模块化规范, node.js 就是参照commonJS规范实现的。commonJS中有一个全局的方法 require()用来加载模块

function myModule(){
  this.hello = function(){
    return "hello"
  }
  this.goodbye = function(){
  return "goodbye"
  }
 }
module.exports = myModule

其实module变量代表当前模块
这样就可以在其他模块中使用这个模块

var myModule = require('myModule');

var myModuleInstance = new myModule();
myModuleInstance.hello();
myModuleInstance.goodbye();

关于commonJS的更多,见CommonJS规范

2. AMD

commonJS定义模块的方式和引入模块的方式还是比较简单的,但不适合浏览器端, 因为commonJS是同步加载的。 而AMD是异步加载的,模块的加载不影响它后面语句的运行。 所有依赖这个模块的语句,都定义在一个回调函数中,等到加载完成之后,这个回调函数才会运行。 这个用require.js实现AMD规范的模块化, 用require.config()指定引用路径等,
通过define()来定义模块, 用requier加载模块

首先我们需要引入require.js文件和一个入口文件main.js. main.js中配置require.config()并规定项目中用到的基础模块。

/** 网页中引入require.js及main.js **/
<script src="js/require.js" data-main="js/main"></script>

/** main.js 入口文件/主模块 **/
// 首先用config()指定各模块路径和引用名
require.config({
  baseUrl: "js/lib",
  paths: {
    "jquery": "jquery.min",  //实际路径为js/lib/jquery.min.js
    "underscore": "underscore.min",
  }
});
// 执行基本操作
require(["jquery","underscore"],function($,_){
  // some code here
});

引用模块的时候,我们将模块名放在[]中作为 require() 的第一参数; 如果我们定义的模块本身也依赖其他模块, 那就需要将他们放在[]中作为define的第一参数

// 定义math.js模块
define(function () {
    var basicNum = 0;
    var add = function (x, y) {
        return x + y;
    };
    return {
        add: add,
        basicNum :basicNum
    };
});
// 定义一个依赖underscore.js的模块
define(['underscore'],function(_){
  var classify = function(list){
    _.countBy(list,function(num){
      return num > 30 ? 'old' : 'young';
    })
  };
  return {
    classify :classify
  };
})

// 引用模块,将模块放在[]内
require(['jquery', 'math'],function($, math){
  var sum = math.add(10,20);
  $("#sum").html(sum);
});

define的第一个参数是依赖的模块, 必须是一个数组。 通过return来暴露接口

通过 require() 来加载模块, 模块的名字默认为模块加载器请求的指定脚本的名字

require(['main'],function(main){
    alert(main.foo());
})

require.js就是根据AMD规范来实现的, 优点是:

  1. 实现js文件的异步加载, 避免网页失去响应
  2. 管理模块之间的依赖性,便于代码的编写和维护
3. CMD

CMD也是异步模块定义
CMD与AMD的区别:
CMD相当于按需加载, 定义一个模块的时候不需要立即制定依赖模块,在需要的时候require就可以了,比较方便。
而AMD则相反,定义模块的时候需要制定依赖模块, 并以形参方式引入回调函数中。

// CMD 按需加载
        define(function(require,exports,module){
            var a = require('./helloWord');
            a.hello(); 

            var b = require('./counter');
            console.log(b.foo); 
            // 2111
            // aaa
        })

        // AMD 定义模块的时候需要制定依赖
        define(['./helloWord','./counter'],function(a,b){
            a.hello();
            console.log(b.foo);
        });
image.png
ES6

ES6在语言规格的层面上,实现了模块功能,而且实现得相当简单, 完全可以取代现有的CommonJS和AMD规范, 成为浏览器和服务器通用的模块解决方案。

ES6模块主要有两个功能: exportimport

export用于对外输出本模块(一个文件可以理解为一个模块)变量的接口

import用于在一个模块中加载另一个含有exoport接口的模块

参考:
https://segmentfault.com/a/1190000010058955
https://juejin.im/post/5aaa37c8f265da23945f365c

https://segmentfault.com/a/1190000004100661
http://es6.ruanyifeng.com/#docs/module

深入系列:
https://github.com/mqyqingfeng/Blog/issues/108
https://zhuanlan.zhihu.com/p/33843378?group_id=947910338939686912

面试题

题目一:ES6与commonJS模块的差异

1. commonJS模块输出的是一个值的拷贝,ES6模块输出的是值的引用。

  • commonJS模块一旦输出一个值,模块内部的变化就影响不到这个值。
  • ES6模块如果使用import从一个模块加载变量,那些变量不会被缓存,而是成为一个指向被加载模块的引用,原始值变了,import加载的值也会跟着变。需要开发者自己保证,真正取值的时候能够取到值。

**2. commonJS 模块是运行时加载, ES6模块是编辑时输出接口

  • 运行时加载: commonJS模块就是对象,即在输入时是加载整个模块,生成一个对象,然后再从整个对象上读取方法,这种加载称为”运行时加载“。 commonJS脚本代码在require的时候,就会全部执行。一旦出现某个模板被”循环加载“,就只能输出已经执行的部分,还未执行的部分不会输出。

  • 编译时加载: ES6模块不是对象,而是通过export命令显式指定输出的代码, import时指定加载某个输出值,而不是加载整个模块,这种加载称为”编译时加载“

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容