在 Node.js 模塊系統中,每一個文件都視爲獨立的模塊,node在運行某個模塊兒時會生成一個module對象node
Module { id: '.', exports: 2, parent: null, filename: '/Users/leinov/github/node-api/module/module.js', loaded: false, children: [ Module { id: '/Users/leinov/github/node-api/module/circle.js', exports: [Object], parent: [Circular], filename: '/Users/leinov/github/node-api/module/circle.js', loaded: true, children: [], paths: [Array] } ], paths: [ '/Users/leinov/github/node-api/module/node_modules', '/Users/leinov/github/node-api/node_modules', '/Users/leinov/github/node_modules', '/Users/leinov/node_modules', '/Users/node_modules', '/node_modules' ] }
node_modules
模塊兒查找路徑,一直查到根目錄Node裏面的模塊系統遵循是CommonJs規範,CommonJs定義的模塊分爲: 模塊標識(module)、模塊定義(exports) 、模塊引用(require),在模塊兒運行的時候都會生成一個module對象和一個exports對象,module對象下也有一個exports對象,默認狀況下這兩個對象都是空對象。若是有引用其餘模塊兒或定義模塊兒 即module.exports或者exports被賦值時,則該模塊兒就是一個有效的帶有返回值的模塊兒。git
一個模塊兒真正導出的是module.exports
的值,exports只是module.exports
的一個引用
能夠簡單理解爲下面這種對象引用和賦值的區別github
let obj1 = {a=1}; let obj2 = obj1; console.log(obj1,obj2); // {a:1} {a:1} obj2.a = 2 console.log(obj1,obj2); // {a:2} {a:2} obj2 = {b:3} console.log(obj1,obj2); // {a:2} {b:3}
obj2只是obj1的一個引用。當 obj2.a
改變時其實改變的是 obj1
和 obj2
都指向的同一個堆裏的數據。但 obj2 ={b:3}
則從新在堆裏開闢了另外一個內存塊兒來存儲。已經跟 obj1
脫離沒有關係了api
因此常常會看到node模塊兒裏會像下面這樣來導出模塊兒。數組
exports = module.exports = ()=>{ do something }
這是爲了讓exports引用指向module.exports同一塊內存,確保數據的一致性。ui
node
文件時同事建立了 module.exports
和 exports
對象exports
是指向 module.exports
的一個引用require("xxx")
其實引用的是xxx
中的 module.exports
而非 exports