NodeJS的模組原理

Ant發表於2020-04-06

最近一直在使用Node JS,在網上看到了一段程式碼我覺得完美的詮釋了Node JS模組載入的原理,其實深究下去,它還詮釋了許多東西:Js模組化程式設計、閉包的真正強大之處等等。閒話不說,先看看這段程式碼:

// - hello.js
void function() {  
    var mapping = {}, cache = {};  
    window.define = function(id, func) {  
        mapping[id] = func  
    };  
    window.require = function(id) {  
        if (!/\.js$/.test(id)) {  
            id += ".js"  
        }  
        if (cache[id]) {  
            return cache[id]  
        } else {  
            return cache[id] = mapping[id]()  
        }  
    }  
}();  
  
define("js/module/hello.js", function(exports) {  
    exports = {};  var name = 'Hello Module'
      
    exports.sayHi = function() {  
        alert("Hi, this is "+ name +" !")  
    };  
      
    return exports  
});   

// - main.js  
var hello = require('js/module/hello');
hello.sayHi(); 
我稍微改寫了一點程式碼,為了能看得更清楚一些,這段程式碼在全域性物件window上繫結了兩個方法:define和require,寫過Node程式碼的人肯定不會陌生,當我需要一個模組的時候,就會使用require來載入它,可能是原生模組例如http, express等,也可能是自己寫的Js檔案模組用相對路徑載入。Node在實現模組載入的時候,就會使用優先本地查詢node_modules目錄,然後一層層往上找,最後再按環境變數在安裝node的目錄執行全域性模組查詢,PS: 自己寫的業務模組一般都使用相對路徑require('./business')來查詢。當Node找到模組之後,第一次載入會比較緩慢,但是一旦載入就會快取起來,類似我們這裡做的cache和map的作用,再查詢就會快很多。所以Node使用require載入模組第一次是阻塞式的,快取之後就應該是非同步非阻塞式的。


我們之所以可以通過require實現Js的模組化程式設計,完全得益於閉包的存在。閉包可以使得我們在傳遞函式呼叫結束,返回函式物件之後,依然可以訪問其中的區域性變數,就像例子中的name一樣,假如它是一個資料庫Connection物件,或者是一個業務Service物件,那麼通過exports,就可以在其他Js檔案中使用它並且程式碼從檔案上是完全隔離的。


這個例子還說明,不止是在服務端,即使在瀏覽器客戶端,我們同樣可以學習Node的原理,做Js的模組化程式設計,當然這也具體要看專案的複雜程度,並非適用所有場景。


相關文章