Node.js | Module System(模組系統)
簡介
為了讓Node.js
的文件可以相互調用,Node.js
提供了一個簡單的模組系統。
模組是Node.js
應用程序的基本組成部分,文件和模組是一一對應的。換言之,一個Node.js
文件就是一個模組,這個文件可能是JavaScript
程式、JSON
或者編譯過的C/C++
擴展。
建立模組
在Node.js
中,建立一個模組非常簡單,如下我們建立一個main.js
,程式如下:
1 | var hello = require('./hello'); |
以上範例中,程式require('./hello')
引入了當前目錄下的hello.js
文件(./
為當前目錄,node.js
默認後綴為js
)。
Node.js
提供了exports
和require
兩個對象,其中exports
是模組公開的接口,require
用於從外部獲取一個模組的接口,即所獲取模組的exports
對象。
接著建立hello.js
,程式如下:
1 | exports.world = function() { |
在以上範例中,hello.js
通過exports
對象把world
作為模組的訪問接口,在main.js
中通過require('./hello')
加載這個模組,然後就可以直接訪問hello.js
中exports
對象的成員函數了。
如果只是想把一個對象封裝到模組中,格式如下:
1 | module.exports = function() { |
例如:
1 | //hello.js |
這樣就可以直接獲得這個對象了:
1 | //main.js |
模組接口的唯一變化是使用module.exports = Hello
代替了exports.world = function(){}
。在外部引用該模組時,其接口對象就是要輸出的 Hello
對象本身,而不是原先的exports
。
服務端的模組放在哪裡
我們已經在程式中使用了模組了。
例如:
1 | var http = require("http"); |
Node.js
中自帶了一個叫做http
的模組,我們在我們的程式中請求它並把返回值賦給一個本地變量。
這把我們的本地變量變成了一個擁有所有http
模組所提供的公共方法的對象。
Node.js
的require
方法中的文件查找策略如下:
由於Node.js
中存在4
類模組(原生模組和3
種文件模組),儘管require
方法極其簡單,但是內部的加載卻是十分複雜的,其加載優先級也各自不同。
如下圖所示:
從文件模組緩存中加載
儘管原生模組與文件模組的優先級不同,但是都會優先從文件模組的緩存中加載已經存在的模組。
從原生模組加載
原生模組的優先級僅次於文件模組緩存的優先級。require
方法在解析文件名之後,優先檢查模組是否在原生模組列表中。以http
模組為例,儘管在目錄下存在一個http/http.js/http.node/http.json
文件,require("http")
都不會從這些文件中加載,而是從原生模組中加載。
原生模組也有一個緩存區,同樣也是優先從緩存區加載。如果緩存區沒有被加載過,則調用原生模組的加載方式進行加載和執行。
從文件加載
當文件模組緩存中不存在,而且不是原生模組的時候,Node.js
會解析require
方法傳入的參數,並從文件系統中加載實際的文件,加載過程中的包裝和編譯細節在前一節中已經介紹過,這裡我們將詳細描述查找文件模組的過程,其中,也有一些細節值得知曉。
require
方法接受以下幾種參數的傳遞:
http
、fs
、path
等,原生模組。./mod
或../mod
,相對路徑的文件模組。/pathtomodule/mod
,絕對路徑的文件模組。mod
,非原生模組的文件模組。
在路徑Y
下執行require(X)
語句執行順序:
1 | 1. 如果 X 是內置模組 |
exports 和 module.exports 的使用
如果要對外暴露屬性或方法,就用 exports 就行,要暴露對象(類似class,包含了很多屬性和方法),就用 module.exports。
註:以上參考了
Node.js模块系统