随着前端js代码复杂度的提升,js模块化是必然趋势,不只好维护,同时依赖很明确,不会全局污染,今天整理一下模块化的几个规范吧~html
script标签引入js文件,相互罗列,可是被依赖的放在前面,不然使用就会报错。以下:前端
<script src="jquery.js"></script>
<script src="jquery_scroller.js"></script>
<script src="main.js"></script>
<script src="other1.js"></script>
<script src="other2.js"></script>
<script src="other3.js"></script>复制代码
即简单的将全部的js文件通通放在一块儿。可是这些文件的顺序还不能出错,好比jquery须要先引入,才能引入jquery插件,才能在其余的文件中使用jquery。缺点很明显:
node
该规范最初是用在服务器端的node的,它有四个重要的环境变量为模块化的实现提供支持:module
、exports
、require
、global
。实际使用时,用module.exports
定义当前模块对外输出的接口(不推荐直接用exports
),用require
加载模块(同步)。jquery
// a-commonJs.js (导出)
var a = 5;var add = function(param){//在这里写上须要向外暴露的函数、变量 return a + param}module.exports.a = a;module.exports.add = add===========================
// b-commonJs.js引用自定义模块,参数包含路径,可省略.js(导入)
var addFn = require('./a-commonJs')console.log(addFn.add(3)) //8console.log(addFn.a) //5复制代码
一点说明: es6
exports 是对 module.exports 的引用。好比咱们能够认为在一个模块的顶部有这句代码: exports = module.exports因此,咱们不能直接给exports赋值,好比number、function等。 复制代码
注意:由于module.exports自己就是一个对象,因此,咱们在导出时可使用 浏览器
module.exports = {foo: 'bar'} //true
缓存
module.exports.foo = 'bar' //true
。bash
可是, exports 是 module.exports 的一个引用,或者理解为exports是一个指针,exports指向module.exports,这样,咱们就只能使用 exports.foo = 'bar'
的方式,而不能使用服务器
exports = {foo: 'bar'} //error 这种方式是错误的,至关于从新定义了exports
babel
一点优势:解决了依赖、全局变量污染的问题
一点缺点: CommonJS用同步的方式加载模块。在服务端,模块文件都存在本地磁盘,读取很是快,因此这样作不会有问题。可是在浏览器端,限于网络缘由,CommonJS不适合浏览器端模块加载,更合理的方案是使用异步加载,好比下边AMD规范。
承接上文,AMD规范则是非同步加载模块,容许指定回调函数,AMD 是 RequireJS 在推广过程当中对模块定义的规范化产出。。
AMD标准中,定义了下面三个API:
require([module], callback)
define(id, [depends], callback)
require.config()
即经过define来定义一个模块,而后使用require来加载一个模块, 使用require.config()指定引用路径。
先到require.js官网下载最新版本,而后引入到页面,以下:
<script data-main="./alert" src="./require.js"></script>
data-main属性不能省略。
以上分别是定义模块,引用模块,运行在浏览器弹出提示信息。
引用模块的时候,咱们将模块名放在[]
中做为reqiure()
的第一参数;若是咱们定义的模块自己也依赖其余模块,那就须要将它们放在[]
中做为define()
的第一参数。
在使用require.js的时候,咱们必需要提早加载全部的依赖,而后才可使用,而不是须要使用时再加载。
一点优势:适合在浏览器环境中异步加载模块、并行加载多个模块
一点缺点:不能按需加载、开发成本大
AMD 推崇依赖前置、提早执行,CMD推崇依赖就近、延迟执行。CMD 是 SeaJS 在推广过程当中对模块定义的规范化产出。
很明显,CMD是按需加载,就近原则。
在ES6中,咱们可使用 import 关键字引入模块,经过 exprot 关键字导出模块,功能较之于前几个方案更为强大,也是咱们所推崇的,可是因为ES6目前没法在浏览器中执行,因此,咱们只能经过babel将不被支持的import编译为当前受到普遍支持的 require。
es6在导出的时候有一个默认导出,export default
,使用它导出后,在import的时候,不须要加上{},模块名字能够随意起。该名字实际上就是个对象,包含导出模块里面的函数或者变量。
可是一个模块只能有一个export default
。
如下引用阮一峰老师的内容:
import
,就会生成一个只读引用。等到脚本真正执行时,再根据这个只读引用,到被加载的那个模块里面去取值。换句话说,ES6 的import
有点像 Unix 系统的“符号链接”,原始值变了,import
加载的值也会跟着变。所以,ES6 模块是动态引用,而且不会缓存值,模块里面的变量绑定其所在的模块。运行时加载: CommonJS 模块就是对象;即在输入时是先加载整个模块,生成一个对象,而后再从这个对象上面读取方法,这种加载称为“运行时加载”。
编译时加载: ES6 模块不是对象,而是经过 export
命令显式指定输出的代码,import
时采用静态命令的形式。即在import
时能够指定加载某个输出值,而不是加载整个模块,这种加载称为“编译时加载”。
CommonJS 加载的是一个对象(即module.exports
属性),该对象只有在脚本运行完才会生成。而 ES6 模块不是对象,它的对外接口只是一种静态定义,在代码静态解析阶段就会生成。
一点说明:本文只是本身的一点笔记,方便本身记忆和理解。\(^o^)/~