Node.js 之 Module 模塊

採用了 Commonjs 規範,經過 module.exports、require 來導出和導入模塊。模塊加載機制中,採用了延遲加載的策略。就是說在用到的狀況下,系統模塊纔會被加載,等加載完成後會放到 binding_cache 中。javascript

分類(模塊類型)

系統模塊

  • 核心模塊(native 模塊),http、buffer、fs 等,底層調用的內建模塊 (C/C++);
  • C/C++ 模塊(built-in 內建模塊),供 native 模塊調用;

第三方模塊

  • 第三方維護的模塊,好比 express、koa、moment.js 等;
  • 本地維護的模塊(以路徑形式的文件模塊)好比 .../ 開頭的;

文件形式

  • javaScript 模塊,module.js
  • json 模塊,module.json
  • C/C++ 模塊,編譯後擴展名爲 .node,module.node

加載機制

加載步驟

經歷 路徑分析文件定位編譯執行java

加載順序

  • 系統緩存,一個模塊被執行後會被緩存起來,提升再次加載速度;
  • 系統模塊,即原生模塊,部分核心模塊已經被編譯成二進制,省略了 路徑分析、文件定位,會直接被加載到了內存中,其中系統模塊定義在源碼的 lib 目錄下;
  • 文件模塊,優先加載 .、..、/ 開頭的,會依次按照 .js、.json、.node 進行擴展名補足嘗試(文件沒有加上擴展名),最好仍是加上文件的擴展名。
  • 目錄模塊,文件模塊加載過程當中,沒有找到,但發現一個一樣的目錄名,就會將這個目錄看成一個來處理。這塊採用了 Commonjs 規範,在文件 package.json 中查找;
  • node_module 模塊,若是系統模塊、路徑文件模塊都找不到,Node.js 會從當前模塊的父目錄開始查找,直到系統的根目錄;

關於緩存問題

模塊緩存後,能夠經過 require.cache 查看已緩存的模塊。node

// 模塊文件 require.module.js
module.exports = {
    name: 'pr',
    say(){ }
}
複製代碼
// 引用模塊文件 require.cache.js
require('./require.module');

console.log('require.cache ----- ');
console.log(require.cache);
複製代碼

對象引用

1.exports 與 module.exports 關係git

const exports =  module.exports;
複製代碼

因此就不能改變 exports 的指向,能夠這樣github

exports.info =  {
    name: 'pr',
    age: 30
}

module.exports = {
    name: 'pr',
    age: 30
}
複製代碼

模塊循環引用

模塊 moduleA.jsmoduleB.js 兩個模塊互相引用,會怎樣?express

// moduleA.js
console.log('模塊 moduleA');
exports.name = 'moduleA name';

age = 27;

const moduleB = require('./moduleB.js');
console.log('moduleA require moduleB =>', moduleB.name);
複製代碼
// moduleB.js
console.log('模塊 moduleB');
exports.name = 'moduleB name';

const moduleA = require('./moduleA.js');
console.log('moduleB require moduleA =>', moduleA.name);
複製代碼

  • 啓動模塊 node moduleA.js,會打印 模塊 moduleA
  • 模塊 moduleA.js 中加載 moduleB.js,打印 模塊 moduleB
  • 模塊 moduleB.js 中又加載 moduleA.js,此時模塊 moduleA.js 尚未執行完,返回模塊 moduleA.jsexports 對象給到模塊 moduleB.js
  • 模塊 moduleB.js 加載完後,其中有個 moduleA.js 中掛載了全局的變量 age,因此能打印出來,最後將模塊 moduleB.jsexports 對象給到模塊 moduleA.js

頗有意思的是,在代碼執行前,會用一個封裝器將執行代碼段封裝起來json

(function(exports, require, module, __filename, __dirname) {
    // something
});
複製代碼

本次代碼 Github緩存

你能夠...

上一篇:Node.js 這一次再也不錯過koa

相關文章
相關標籤/搜索