JS模块化

模块

  • 函数式
function fn(){...}

模块是实现特定功能的文件,将多个函数编写在同一个文件中就构成了一个模块,加载文件即可调用文件中的函数。其缺点是污染全局变量,无法保证与其他模块文件发生命名冲突,且模块成员之间没有什么关系。

  • 对象式
// 将模块作为一个对象,模块成员封装在对象中。
var module = {
  prop:0,
  fn:function(){...}
};
// 通过调用对象属性访问使用模块成员
module.fn();
module.prop = 1;// 缺点是暴露了模块成员,外部可修改模块内部状态。

将模块作为一个对象,模块成员封装在对象中,通过调用对象属性访问使用模块成员。缺点是暴露了模块成员,外部可修改模块内部状态。

  • 自执行函数
var module = (function(){
  var prop = 0; // 私有变量
  var fn = function(){
    console.log('fn');
  };
  return {fn:fn};
})();
module.fn();// fn
// 外部无法访问内部私有变量
console.log(module.prop); // undefined

立即执行函数的优点在于外部无法访问内部私有变量。

模块化

JS模块化起源于Node.js,前端模块化规范分为CommonJS、AMD、CMD,CommonJS使用在服务端,AMD和CMD使用在浏览器环境。

  • AMD(Asynchronous Module Definition)异步模块定义
  • CMD(Common Module Definition)通用模块定义

前端模块化规范

  • CommonJS Module/2.0 规范
  • AMD 是RequireJS在推广过程中对模块化定义的规范化产出,特点是提前执行(异步加载即依赖先执行)、延迟执行。(异步加载,依赖前置,提前执行)
    • define 定义模块 define(['require', 'foo'], function(){return})
    • require 加载模块依赖前置 require(['foo', 'bar'], function(foo,bar){})
  • CMD 是SeaJS在推广过程中对模块化规范化产生,特点是延迟执行,即运行时按需加载,顺序执行。(同步加载,依赖就近,延迟执行)
    • define 定义 exports 导出 define(function(require, exports, module){...})
    • module 上存储了当前模块上的一些对象
    • require(filepath) 直接引入,require.async异步引入。

CommonJS

CommonJS是服务器端模块的规范,由Node.js推广使用。由于服务端编程复杂性,若没有模块将很难与操作系统及其他引用互动。CommonJS规范指出:

  • 一个单独的文件就是一个模块
    每个模块都有一个独立的作用域,在模块内部定义的变量,无法被其他模块读取,除非定义为global对象的属性。
  • 输出模块变量的方式是使用module.exports对象
  • 加载模块使用require()方法,该方法读取一个文件并执行,返回文件内部的module.exports对象。
# math.js 计算参数的和
exports.add = function(){
  var sum = 0;
  var i = 0;
  while(i < arguments.length;){
    sum += arguments[i++];
  }
  return sum;
}
# increment.js 累加计算
var add = require('math').add;
exports.inc = function(val){
  return add(val, 1);
}
# test.js
var increment = require('increment').inc;
console.log(increment(1));// 2

CommonJS中require()是同步的,模块系统需同步读取模块文件内容,并编译执行以获得模块接口。然后,这在浏览器端问题多多。由于浏览器端加载JS最佳的方式是在document中插入<script>标签,但<script>脚本标签天生是异步加载的。所以传统的CommonJS模块在浏览器环境中无法正常加载。

解决的思路是开发服务端组件,对模块代码做静态分析,将模块与其依赖一并返回给浏览器。但是这需要服务端安装额外的组件并因此要调整一系列底层架构。

另一种解决的方案是用一套标准模板来封装模块定义,这套模块代码为模块加载器提供了机会,使其能再模块代码执行之前,对代码模块进行静态分析,并动态生成依赖列表。

define(function(require, exports, module){
  // the module code goes here...
})

上述案例修改

# math.js
define(function(require, exports, module){
  exports.add = function(){
    var sum=0, i=0;
    while(i < argument.length){
      sum  += argument[i++]; 
    }
    return sum;
  }
})
# increment.js
define(function(require, exports, module){
  var add = require('math').add;
  exports.inc = function(val){
    return add(val, 1);
  }
})
# test.js
define(function(require, exports, module){
  var inc = require('increment').inc;
  inc(1);// 2
});

AMD

AMD异步模块定义,由于JS原生不支持,使用AMD规范进行页面开发需使用对应的库函数,也就是大名鼎鼎的RequireJS,实际上AMD是RequireJS在推广过程中对模块定义的规范化产出。
RequireJS采用异步加载模块,模块加载不影响后续语句的执行。所有依赖此模块的语句都定义在一个回调函数中,等待加载完毕后回调函数才开始执行。

RequireJS主要解决了两个问题

  • 多个JS文件可能具有依赖关系,被依赖的文件需早于依赖它的文件加载到浏览器。
  • JS加载时浏览器会停止页面渲染,加载文件越多页面失去响应的时间将越长。

RequireJS依赖前置

# main.js
// 定义别名
requirejs.config({
  paths:{
    jquery:'jquery.min'// 可省略后缀.js
  }
});

// 引入模块,使用变量$表示jquery模块
requirejs(['jquery'],function($){
  $('body').css('background-color', 'red');
});

require.config用于定义别名,在paths属性下配置别名,然后通过requirejs传入所需引入的模块名。

requirejs通过define()方法定义模块,在模块内的方法和属性外部是无法访问的,只能通过return返回才行。

# math.js 定义模块并引入jquery
define('math', ['jquery'], function($){
  return {
    add:function(x,y){
      return x+y;
    }
  };
});

引入模块也可写成require(),使用时首先将math.jsmain.js中引入。

require(['jquery', 'math'], function($, math){
  console.log(math.add(1, 2));// 3
})

RequireJS 定义模块

RequireJS定义了define()函数,它是一个全局变量用于定义模块。

define(id?, dependencies?, factory);
# eg:
define(['require', 'foo'], function(){return})
  • id 可选,指定模块的名字。若无则模块默认为模块加载器请求的指定脚本的名字。若有则模块名必须是顶级的和绝对的。
  • dependencies 可选,当前模块依赖,已被模块定义的模块标识的数组字面量。若无则默认为["require", "exports", "module"],若工厂方法长度小于3则加载器会选择以函数的长度属性指定的参数个数调用工厂方法。
  • factory 工厂方法,模块初始化要执行的函数或对象,若为函数则应只被执行一次,若为对象则为模块的输出值。

例如

define('alpha', ['require','exports','beta'], function(require, exports, beta){
  export.verb = function(){
    return beta.verb();
    // return require('beta').verb();
  }
})

RequireJS 加载模块(依赖前置)

RequireJS采用require()语句加载模块,不同于CommonJS,它要求两个参数。

require([modules], callback)
# eg
require(['foo', 'bar'], function(foo,bar){})
  • module 要加载的模块数组
  • callback 加载成功后的回调函数

例如

# test.js
require(['increment'], function(inc){
  inc.add(1);
});

CMD

CMD通用模块定义,其规范是国内发展出来的,就像AMD有个RequireJS,CMD有个浏览器的实现叫作SeaJS,SeaJS要解决的问题和RequireJS一样,只不过在模块定义方法和模块加载时机上有所不同。

CMD规范中一个模块就是一个文件

define(function(require, exports, module){
  // goes here
})
  • require 是将其他模块导入的参数
  • exports 是将模块内的属性和方法导出的参数
  • module 是一个对象,存储了与当前模块相关联的属性和方法。

AMD和CMD的区别

  • AMD是依赖关系前置,即在定义模块时就要声明其依赖的模块。
  • CMD是按需加载依赖(依赖就近),即只有使用某个模块时再去require
# CMD 依赖就近
define(function(require, exports, module){
  var a= require('a');
  a.todo();
  ...
  var b= require('b');
  b.todo();
});

// AMD 依赖前置
define(['a','b'], function(a,b){
  a.todo();
  b.todo();
})

SeaJS

https://github.com/seajs/seajs

$ npm i -g npm
$ npm install -g bower
$ mkdir seajs && cd seajs
$ bower install seajs
$ bower install jquery

$ vim index.html
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>seajs</title>
    <script src="bower_components/seajs/dist/sea.js"></script>
</head>
<body>
    <script type="text/javascript">
    seajs.config({
        base:'./bower_components/jquery/dist/',
        alias:{
            'jquery':'jquery.js'
        }
    });
    seajs.use('./js/main');
    </script>
</body>
</html>

$ vim js/main.js
define(function(require, exports, module){
    require('jquery');
});

RequireJS

http://requirejs.org/

$ mkdir requirejs && cd requirejs
$ bower install requirejs

$ vim index.html
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>requirejs</title>
</head>
<body>
    <script type="text/javascript" src="bower_components/requirejs/require.js" data-main="js/app.js"></script>
</body>
</html>

$ vim js/app.js
define(['require','app'], function(){
    require(['test'],function(test){
        test.init();
    });
});

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

推荐阅读更多精彩内容

  • 什么是模块化? 简单理解:将各个功能封装为独立的模块,当需要某个功能时,只需要加载相应的模块即可 为什么出现模块化...
    放风筝的小小马阅读 848评论 0 6
  • 历史上JavaScript一直没有模块体系,在其他高级语言中,Java有类文件,Python有import机制,R...
    还是那个西瓜阅读 264评论 0 1
  • 什么是模块化? 1,把你的代码写成模块的2,把这单独的模块写在单独的文件里 模块化有什么好处? 1,让代码更清晰2...
    PYFang阅读 377评论 0 0
  • 什么是 JavaScript 模块? JavaScript 模块允许我们把项目中的代码分散成一个个单独的文件,或者...
    Daniel_Y阅读 815评论 0 0
  • CommonJS CommonJS规范是诞生比较早的。NodeJS就采用了CommonJS。是这样加载模块: 这种...
    baxiamali阅读 188评论 0 0