谨以此文记录了解js模块的过程

随着ES6的出现,js模块已经成为正式的标准了。曾经为了解决js模块问题而发展起来的民间秘籍,requireJs(AMD)、SeaJs(CMD)、Node(CommonJs),已经或者不久的将来会成为历史。了解历史也是很重要的,因为正式标准就是以民间秘籍为基础而发展起来的,有些规范仍然被广泛应用于开发中(CommonJS)。再者,进入有些公司,你的工作可能是继承前辈们的代码,为了不至于始终一脸懵,认识一下这些东西是很重要的。

关于js模块的发展历史

github上玉伯发表的issue:地址

CommonJs模块规范

  • 用于同步加载模块
  • 主要应用于服务端

关于CommonJS的说明:commonjs

Node参考CommonJS实现了一套模块系统,npm上有大量的CommonJS的包,ECMAScript取代CommonJS可能还需要一些时日。

CommonJs对模块的定义很简单,主要分为模块引用、模块定义和模块标识。

模块引用

使用require()方法,该方法接收模块标识,引入一个模块的API到当前上下文。

var math = require("math");
math.add();

模块定义

上下文提供exports对象用于导出当前模块的方法或者变量,并且它是唯一导出的出口。module对象,代表模块自身,exports是module的属性。将方法挂载在exports对象上作为属性即可定义导出的方式。

exports.add = function () {
var sun = 0,
i = 0,
args = arguments,
l = args.length; while (i < 1) {
sum += args[i++];
}
return sum;
}

模块标识

模块标识是传递给require()方法的参数,它必须是符合小驼峰命名的字符串,或者以.,..开头的相对路径,或者绝对路径。它可以没有文件名后缀.js。

优势

  1. 语法简单,只需要定义module.exports属性,剩下的模块代码与标准Javascript无差异。引用模块的方法也很简单,只需要使用require函数
  2. CommonJS是Node.js默认的模板格式,所以我们可以是用成千上万的包

缺点

  1.  最大的缺点就是不显示的支持浏览器,(浏览器端的javascript不支持module变量和export属性,当然也有方法可以在浏览器中使用CommonJS模块,可以查看这里)由此也产生了另外的用于前端浏览器的规范AMD

AMD与requireJS

AMD是CommonJS的一个延伸,制定了能够在前端应用的模块规范。

  • 异步加载模块
  • 主要用于浏览器

github中AMD的定义——地址

规范

// 模块定义
// id和dependencies都是可选的
// factory是模块的实际代码
define(id, dependencies, factory); define("alpha", ["require", "exports", "beta"], function (require, exports, beta) {
exports.verb = function () {
return beta.verb();
//Or:
return require("beta").verb();
}
});
// 不带id(名称)
define(["alpha"], function (alpha) {
return {
verb: function () {
return alpha.verb() + 2;
}
};
});
// 不带名称和依赖
define({
add: function (x, y) {
return x + y;
}
}); // 引用模块
// module是要加载的模块
// callback在加载完成之后运行
require([module], callback)
require(['a', 'b'], function (a, b) {
//modules a and b are now available for use.
});

优势

  1. 自动处理依赖,无序考虑引入顺序
  2. 异步加载模块,避免阻塞

requireJS

requireJS实现了AMD规范,具体使用方法请移步——中文API

CMD与SeaJS

CMD是玉伯在学习requireJs的过程中,自己提出的一个规范。SeaJs作为实现该规范的工具,一度成为与requireJs并肩的工具(当然主要是在国内)

玉伯在2015年宣布Seajs已死。ES2015正式发布了。不再需要民间力量修补这个漏洞了。

ES6Module

以上所有规范都是ES6之前产生的,为ES6奠定了基础。ES6汲取了CommonJS和AMD的优点:

  • 与CommonJS类似,语法简单,一个文件是一个模块
  • 与AMD类似,同样支持异步加载模块

语法

http://es6.ruanyifeng.com/#docs/module

总结

CommonJS规范,语法简单、同步加载、主要应用于服务端。目前在Node中仍然占据重要地位。

AMD规范,作为CommonJS的延伸,语法略微复杂一些,异步加载,主要应用于浏览器。

ES6 Module作为javascript内置的模块标准,虽然浏览器目前并不完全支持,但是有Babel等转换器,帮助转换ES6语法到ES5语法,从现在开始学习并应用ES6,也许是比较好的选择。

参考

  1. js模块发展历史——地址
  2. 知乎上玉伯关于CMD和AMD的区别的回答——地址
  3. require.js——中文官网
  4. 《深入理解es6》
  5. exploringJs——https://exploringjs.com/es6/

javascript的模块发展的更多相关文章

  1. 研究一下javascript的模块规范(CommonJs/AMD/CMD)

    最近写react需要使用nodejs作为开发环境,需要通过npm安装一些第三方的依赖库,因此慢慢感觉到nodejs基础薄弱对我带来了一些不安全感,尤其是javascript模块这一块听到了很多概念,比 ...

  2. seajs实现JavaScript 的 模块开发及按模块加载

    seajs实现了JavaScript 的 模块开发及按模块加载.用来解决繁琐的js命名冲突,文件依赖等问题,其主要目的是令JavaScript开发模块化并可以轻松愉悦进行加载. 官方文档:http:/ ...

  3. JavaScript中模块“写法”

    在JavaScript模块到底是什么 event = function() { // do more return { bind: function() {}, unbind: function() ...

  4. 在浏览器中高效使用JavaScript module(模块)

    在浏览器中也可以使用JavaScript modules(模块功能)了.目前支持这一特性的浏览器包括: Safari 10.1. 谷歌浏览器(Canary 60) – 需要在chrome:flags里 ...

  5. python与javascript 引入模块的方法对比

    1.引入整体模块对比 python 方法一: # 引入全部函数 from xxx import * # 直接使用模块里面的各函数或者属性 test() 方法二: # 引入全局的模块 import gl ...

  6. javascript 异步模块加载 简易实现

    在javascript是没有类似java或其他语言的模块概念的,因此也不可能通过import或using等关键字来引用模块,这样造成了复杂项目中前端代码混乱,变量互相影响等. 因此在复杂项目中引入AM ...

  7. JavaScript AMD 模块加载器原理与实现

    关于前端模块化,玉伯在其博文 前端模块化开发的价值 中有论述,有兴趣的同学可以去阅读一下. 1. 模块加载器 模块加载器目前比较流行的有 Requirejs 和 Seajs.前者遵循 AMD规范,后者 ...

  8. [JavaScript] 前端模块加载简单实现(require)

    模块加载的简单实现 (function(win) { var baseUrl; var paths; var script_cache = {}; var script_queue = []; var ...

  9. [JavaScript] 前端模块编程实现

    前端模块化 前端早期写代码都是全局变量满天飞,这种情况会造成全局命名空间污染,变量冲突等问题 var a = 1; var b = 2; function c(){} function d(){} 后 ...

随机推荐

  1. ffmpeg捕捉摄像头发送rtmp

    打印 DirectShow 支持的设备列表(true 可用1替换): ffmpeg -list_devices true -f dshow -i dummy 本计算机打印出的信息如下:[dshow @ ...

  2. Gym - 100851G:Generators(人尽皆知但是WA题)

    题意:现在有函数,每一项Xi=(A*X(i-1)+B)%C.现在给定N个函数以及K:X0,A,B,C.然你再每个函数选择一个数,使得其和最大,而且不被K整除. X0,A,B,C<=1e3 :K& ...

  3. OSS阿里云文件上传 demo。

    所需jar包: aliyun-openservices-1.2.3.jar jdom-1.1.jar commons-codec-1.4.jar commons-logging-1.1.1.jar g ...

  4. ACM学习历程—HDU5587 Array(数学 && 二分 && 记忆化 || 数位DP)(BestCoder Round #64 (div.2) 1003)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5587 题目大意就是初始有一个1,然后每次操作都是先在序列后面添加一个0,然后把原序列添加到0后面,然后 ...

  5. bzoj 2553: [BeiJing2011]禁忌 AC自动机+矩阵乘法

    题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=2553 题解: 利用AC自动机的dp求出所有的转移 然后将所有的转移储存到矩阵中,进行矩阵 ...

  6. Tensorflow Summary用法

    本文转载自:https://www.cnblogs.com/lyc-seu/p/8647792.html Tensorflow Summary用法 tensorboard 作为一款可视化神器,是学习t ...

  7. 三种 Failover 之 Client-Side Connect time Failover、Client-Side TAF、Service-Side TAF

    三种 Failover 之 Client-Side Connect time Failover.Client-Side TAF.Service-Side TAF 理论背景 Oracle  RAC 同时 ...

  8. Python:itertools库的使用

    转于:https://blog.csdn.net/neweastsun/article/details/51965226 博主:neweastsun的专栏 介绍 itertools是python内置的 ...

  9. MATLAB模糊逻辑工具箱函数

    说明:本文档中所列出的函数适用于Matlab5.3以上版本,为了简明起见,只列出了函数名,若需要进一步的说明,请参阅MATLAB的帮助文档. 1. GUI工具 Anfisedit      打开ANF ...

  10. 在python3.6下 发明一个类似python3.7 dataclass数据类,不用在 __init__中self.xx

    虽然我用3.6,但我在2.7转3.6时候,把3.3 3.4 3.5 3.6的变化都看了一次,虽然已经忘了哪些变化.同时也关注3.7 3.8的变化,3.7中就有1个数据类印象深刻,因为之前在定义这种类时 ...