一普筹、CommonJS
????????Node.js是commonJS規(guī)范的主要實(shí)踐者粉捻,它有四個(gè)重要的環(huán)境變量為模塊化的實(shí)現(xiàn)提供支持:module、exports斑芜、require肩刃、global。實(shí)際使用時(shí)杏头,用module.exports定義當(dāng)前模塊對(duì)外輸出的接口(不推薦直接用exports)盈包,用require加載模塊。
// 定義模塊math.js
var basicNum = 0;
function add(a, b) {
return a + b;
}
module.exports = { //在這里寫(xiě)上需要向外暴露的函數(shù)醇王、變量
add: add,
basicNum: basicNum
}
// 引用自定義的模塊時(shí)呢燥,參數(shù)包含路徑寓娩,可省略.js
var math = require('./math');
math.add(2, 5);
// 引用核心模塊時(shí)叛氨,不需要帶路徑
var http = require('http');
http.createService(...).listen(3000);
????????commonJS用同步的方式加載模塊。在服務(wù)端棘伴,模塊文件都存在本地磁盤(pán)寞埠,讀取非常快焊夸,所以這樣做不會(huì)有問(wèn)題仁连。但是在瀏覽器端,限于網(wǎng)絡(luò)原因阱穗,更合理的方案是使用異步加載饭冬。
二、AMD和require.js
????????AMD規(guī)范采用異步方式加載模塊揪阶,模塊的加載不影響它后面語(yǔ)句的運(yùn)行昌抠,所有依賴這個(gè)模塊的語(yǔ)句,都定義在一個(gè)回調(diào)函數(shù)中鲁僚,等到加載完成之后炊苫,這個(gè)回調(diào)函數(shù)才會(huì)運(yùn)行,這里介紹用require.js實(shí)現(xiàn)AMD規(guī)范的模塊化蕴茴,用require.config()指定引用路徑等劝评,用define()定義模塊,用require()加載模塊倦淀;
????????首先我們需要引入require.js文件和一個(gè)入口文件main.js蒋畜,main.js中配置require.config()并規(guī)定項(xiàng)目中用到的基礎(chǔ)模塊;
/** 網(wǎng)頁(yè)中引入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", //實(shí)際路徑為js/lib/jquery.min.js
"underscore": "underscore.min",
}
});
// 執(zhí)行基本操作
// 引用模塊的時(shí)候撞叽,我們將模塊名放在[]中作為reqiure()的第一參數(shù)姻成;如果我們定義的模塊本身也依賴其他模塊,那就需要將它們放在[]中作為define()的第一參數(shù);
require(["jquery","underscore"],function($,_){
// some code here
});
// 定義math.js模塊
define(function () {
var basicNum = 0;
var add = function (x, y) {
return x + y;
};
return {
add: add,
basicNum :basicNum
};
});
// 定義一個(gè)依賴underscore.js的模塊
define(['underscore'],function(_){
var classify = function(list){
_.countBy(list,function(num){
return num > 30 ? 'old' : 'young';
})
};
return {
classify :classify
};
})
// 引用模塊插龄,將模塊放在[]內(nèi)
require(['jquery', 'math'],function($, math){
var sum = math.add(10,20);
$("#sum").html(sum);
});
三、CMD
????????require.js在申明依賴的模塊時(shí)會(huì)在第一之間加載并執(zhí)行模塊內(nèi)的代碼:CMD是另一種js模塊化方案科展,它與AMD很類似均牢,不同點(diǎn)在于:AMD 推崇依賴前置、提前執(zhí)行才睹,CMD推崇依賴就近徘跪、延遲執(zhí)行。此規(guī)范其實(shí)是在sea.js推廣過(guò)程中產(chǎn)生的琅攘。
/** AMD寫(xiě)法 **/
define(["a", "b", "c", "d", "e", "f"], function(a, b, c, d, e, f) {
// 等于在最前面聲明并初始化了要用到的所有模塊
a.doSomething();
if (false) {
// 即便沒(méi)用到某個(gè)模塊 b垮庐,但 b 還是提前執(zhí)行了
b.doSomething()
}
});
/** CMD寫(xiě)法 **/
define(function(require, exports, module) {
var a = require('./a'); //在需要時(shí)申明
a.doSomething();
if (false) {
var b = require('./b');
b.doSomething();
}
});
/** sea.js **/
// 定義模塊 math.js
define(function(require, exports, module) {
var $ = require('jquery.js');
var add = function(a,b){
return a+b;
}
exports.add = add;
});
// 加載模塊
seajs.use(['math.js'], function(math){
var sum = math.add(1+2);
});
四、ES6
????????ES6在語(yǔ)言標(biāo)準(zhǔn)的層面上坞琴,實(shí)現(xiàn)了模塊功能哨查,而且實(shí)現(xiàn)得相當(dāng)簡(jiǎn)單,旨在成為瀏覽器和服務(wù)器通用的模板解決方案剧辐,其模塊功能主要由兩個(gè)命令構(gòu)成:export和import寒亥,export命令用于規(guī)定模塊的對(duì)外接口,import命令用于輸入其他模塊提供的功能荧关;使用import命令的時(shí)候溉奕,用戶需要知道所要加載的變量名或函數(shù)名,其實(shí)ES6還提供了export default命令羞酗,為模塊指定默認(rèn)輸出腐宋,對(duì)應(yīng)的import語(yǔ)句不需要使用大括號(hào),這也更趨近于ADM的引用寫(xiě)法檀轨;ES6的模塊不是對(duì)象,import命令會(huì)被 JavaScript 引擎靜態(tài)分析欺嗤,在編譯時(shí)就引入模塊代碼参萄,而不是在代碼運(yùn)行時(shí)加載,所以無(wú)法實(shí)現(xiàn)條件加載煎饼。也正因?yàn)檫@個(gè)讹挎,使得靜態(tài)分析成為可能;
/** 定義模塊 math.js **/
var basicNum = 0;
var add = function (a, b) {
return a + b;
};
export { basicNum, add };
/** 引用模塊 **/
import { basicNum, add } from './math';
function test(ele) {
ele.textContent = add(99 + basicNum);
}
/** export default **/
//定義輸出
export default { basicNum, add };
//引入
import math from './math';
function test(ele) {
ele.textContent = math.add(99 + math.basicNum);
}
五吆玖、 ES6 模塊與 CommonJS 模塊的差異
1. CommonJS 模塊輸出的是一個(gè)值的拷貝筒溃,ES6 模塊輸出的是值的引用。
- CommonJS 模塊輸出的是值的拷貝沾乘,也就是說(shuō)怜奖,一旦輸出一個(gè)值,模塊內(nèi)部的變化就影響不到這個(gè)值翅阵。
- ES6 模塊的運(yùn)行機(jī)制與 CommonJS 不一樣歪玲。JS 引擎對(duì)腳本靜態(tài)分析的時(shí)候迁央,遇到模塊加載命令
import
,就會(huì)生成一個(gè)只讀引用滥崩。等到腳本真正執(zhí)行時(shí)岖圈,再根據(jù)這個(gè)只讀引用,到被加載的那個(gè)模塊里面去取值钙皮。換句話說(shuō)蜂科,ES6 的import
有點(diǎn)像 Unix 系統(tǒng)的“符號(hào)連接”,原始值變了短条,import
加載的值也會(huì)跟著變导匣。因此,ES6 模塊是動(dòng)態(tài)引用慌烧,并且不會(huì)緩存值逐抑,模塊里面的變量綁定其所在的模塊。
2. CommonJS 模塊是運(yùn)行時(shí)加載屹蚊,ES6 模塊是編譯時(shí)輸出接口厕氨。
運(yùn)行時(shí)加載: CommonJS 模塊就是對(duì)象;即在輸入時(shí)是先加載整個(gè)模塊汹粤,生成一個(gè)對(duì)象命斧,然后再?gòu)倪@個(gè)對(duì)象上面讀取方法,這種加載稱為“運(yùn)行時(shí)加載”嘱兼。
編譯時(shí)加載: ES6 模塊不是對(duì)象国葬,而是通過(guò)
export
命令顯式指定輸出的代碼,import
時(shí)采用靜態(tài)命令的形式芹壕。即在import
時(shí)可以指定加載某個(gè)輸出值汇四,而不是加載整個(gè)模塊,這種加載稱為“編譯時(shí)加載”踢涌。
CommonJS 加載的是一個(gè)對(duì)象(即module.exports
屬性)通孽,該對(duì)象只有在腳本運(yùn)行完才會(huì)生成。而 ES6 模塊不是對(duì)象睁壁,它的對(duì)外接口只是一種靜態(tài)定義背苦,在代碼靜態(tài)解析階段就會(huì)生成