CommonJS規(guī)范
CommonJS是一種思想冯遂,它的終極目標是使應用程序開發(fā)者根據(jù)CommonJS API編寫的JavaScript應用可以在不同的JavaScript解析器和HOST環(huán)境上運行越败。目前,有四大平臺支持CommonJS API:Rhino赁温、Spidermonkey、v8鳞绕、JavaScriptCore氧急。
Node.js是一個事件驅動I/O服務端JavaScript環(huán)境,基于Google的V8引擎
引入
- 模塊的引入大大提高了代碼的可維護性并避免函數(shù)名和變量名沖突
- 在node.js中文件和模塊是一一對應的八回,即每個文件被視為一個獨立的模塊酷愧;模塊分為兩種類型:原生模塊和文件模塊
- Node提供了exports和require兩個對象,其中exports是模塊公開的接口,require用于從外部獲取一個模塊接口,即所獲取模塊的exports對象
原生模塊(核心模塊)
原生模塊在Node.js源代碼編譯的時候編譯進了二進制執(zhí)行文件驾诈,加載的速度最快
在這不具體介紹,有興趣可參考https://itbilu.com/nodejs/core/N1tv0Pgd-.html
文件模塊
文件模塊中伟墙,又分為3類模塊翘鸭。這三類文件模塊以后綴來區(qū)分,Node.js會根據(jù)后綴名來決定加載方法戳葵。
.js。通過fs模塊同步讀取js文件并編譯執(zhí)行汉匙。
.node拱烁。通過C/C++進行編寫的Addon。通過dlopen方法進行加載噩翠。
.json戏自。讀取json文件,調用JSON.parse解析加載伤锚。
在路徑 Y 下執(zhí)行 require(X) 語句執(zhí)行順序:
1. 如果 X 是一個核心模塊擅笔,
a. 返回核心模塊
b. 結束
2. 如果 X 是以 '/' 開頭(絕對路徑)
a. 設 Y 為文件系統(tǒng)根目錄
3. 如果 X 是以 './' 或 '/' 或 '../' 開頭(相對路徑)
a. 加載文件(Y + X)
b. 加載目錄(Y + X)
4. 加載Node模塊(X, dirname(Y))
5. 拋出 "未找到"
加載文件(X)
1. 如果 X 是一個文件,加載 X 作為 JavaScript 文本屯援。結束
2. 如果 X.js 是一個文件猛们,加載 X.js 作為 JavaScript 文本。結束
3. 如果 X.json 是一個文件狞洋,解析 X.json 成一個 JavaScript 對象弯淘。結束
4. 如果 X.node 是一個文件,加載 X.node 作為二進制插件吉懊。結束
加載索引(X)
1. 如果 X/index.js 是一個文件庐橙,加載 X/index.js 作為 JavaScript 文本。結束
3. 如果 X/index.json 是一個文件借嗽,解析 X/index.json 成一個 JavaScript 對象态鳖。結束
4. 如果 X/index.node 是一個文件,加載 X/index.node 作為二進制插件恶导。結束
加載目錄(X)
1. 如果 X/package.json 是一個文件浆竭,
a. 解析 X/package.json,查找 "main" 字段
b. let M = X + (json main 字段)
c. 加載文件(M)
d. 加載索引(M)
2. 加載索引(X)
加載Node模塊(X, START)
1. let DIRS=NODE_MODULES_PATHS(START)
2. for each DIR in DIRS:
a. 加載文件(DIR/X)
b. 加載目錄(DIR/X)
NODE_MODULES_PATHS(START)
1. let PARTS = path split(START)
2. let I = count of PARTS - 1
3. let DIRS = []
4. while I >= 0,
a. if PARTS[I] = "node_modules" CONTINUE
b. DIR = path join(PARTS[0 .. I] + "node_modules")
c. DIRS = DIRS + DIR
d. let I = I - 1
5. return DIRS
require
require函數(shù)用于在當前模塊中加載和使用別的模塊甲锡,傳入一個模塊名兆蕉,返回一個模塊導出對象。require方法接受以下幾種參數(shù)的傳遞:
http缤沦、fs虎韵、path等。原生模塊缸废。(優(yōu)先載入)
./mod或../mod包蓝。相對路徑的文件模塊驶社。
/a/mod,絕對路徑的文件模塊测萎。
mod亡电,非原生模塊的文件模塊。
export
module.exports與exports
每一個node.js執(zhí)行文件硅瞧,都自動創(chuàng)建一個module對象份乒,同時,module對象會創(chuàng)建一個叫exports的屬性腕唧,初始化的值是 {}
exports是引用 module.exports的值或辖。module.exports 被改變的時候,exports不會被改變枣接,而模塊導出的時候颂暇,真正導出的執(zhí)行是module.exports,而不是exports
exports在module.exports 被改變后但惶,失效耳鸯。
通過exports = module.exports的方法,讓其恢復原來的特點膀曾。
/*
module.exports.example1=function () {
console.log(1);
}
exports = module.exports;
*/
exports.a=function () {
console.log('a');
}
module.exports.a = function () {
console.log(2)
}
module.exports是真正的接口县爬,exports只不過是它的一個輔助工具〖酥 最終返回給調用的是Module.exports而不是exports捌省。
所有的exports收集到的屬性和方法,都賦值給了module.exports碉钠,而前提是module.exports本身不具備任何屬性和方法纲缓。如果,Module.exports已經具備一些屬性和方法喊废,那么exports收集來的信息將被忽略祝高。