摘要:現在,我們可以開始探討介面的設計模式了。匯出命名空間一個簡單且常用的設計模式就是匯出一個包含數個屬性的物件,這些屬性具體的內容主要是函式,但並不限於函式。如此,我們就能夠透過匯入該模組來取得這個命名空間下一系列相關的功能。
前言
這篇文章試著要整理,翻譯Export This: Interface Design Patterns for Node.js Modules這篇非常值得一讀的文章。
但因為這篇文章有些時日了,部分範例已經不符合現況。故這是一篇加上小弟收集彙整而成的更新翻譯。
當你在 Node 中載入一個模組,我們到底會取回什麼?當我們撰寫一個模組時我們又有哪些選擇可以用來設計程式的介面?
在我第一次學習 Node 的時候,發現在 Node 中有太多的方式處理這個問題,由於 Javascript 本身非常彈性,加上在社群中的開發者們各自都有不同的實作風格,這讓當時的我感到有點挫折。
在原文作者的學習旅程中曾持續的觀察尋找好的方式以應用在其的工作上,在這篇文章中將會分享觀察到的 Node 模組設計方式。
大略總結了 7 種設計模式(pattern)
匯出命名空間 Namespace
匯出函式 Function
匯出高階函式 High-Order Function
匯出建構子/構建函式 Constructor
匯出單一實例物件 Singleton
擴展全域物件 Extend Global Object
套用猴子補丁 Monkey Patch
require, exports 和 module.exports首先我們需要先聊點基礎的知識
在 Node 官方文件中定義了匯入一個檔案就是匯入一個模組。
In Node.js, files and modules are in one-to-one correspondence. - Node 文件
每個模組內都有一個隱式(implicit)的 module 物件,這個物件身上有一個屬性 exports (即 module.exports)。當我們使用 require() 時它所傳回的,即是這個模組的 module.exports 所指向的東西。
每個模組的 module.exports 預設都是一個空物件 {},而每個模組內帶有一個 exports 捷徑變數,預設指向 module.exports。
在預設情況下,module.exports 是一個物件,而 exports 又指向該物件,所以在預設情況下,以下兩個操作視為等效:
exports.foo = "hello"; // 等同於 module.exports.foo = "hello";
如果,你重新指定了某東西給 module.exports,當然 module.exports 跟預設物件的關係就因此被打斷,而此時 exports 捷徑變數仍然是指向預設物件的。前面有提到,當我們使用 require() 時,模組是以 module.exports 所指向的東西而揭露給外部,因此模組的介面是由 module.exports 所決定,與 exports 捷徑變數無關。
因此,一旦 module.exports 被指定新值後,使用捷徑變數 exports 所掛入預設物件的內容都將無法被揭露,因而可視為無效。
為了讓您更好理解關於 exports 與 module.exports 下面的範例提供了較詳細的說明
var a = { id: 1 } var b = a console.log(a) // {id: 1} console.log(b) // {id: 1} // b 參考指向 a,意味著修改 b 的屬性 a 會跟著變動 b.id = 2 console.log(a) // {id: 2} console.log(b) // {id: 2} // 但如果將一個全新的物件賦予 b 那麼參考的關係將會中斷 b = { id: 3 } console.log(a) // {id: 2} console.log(b) // {id: 3}
另外比較具體的範例
// 模組預設: module.exports = {}, 而 exports = module.exports, 等同於指向預設的空物件 /* person.js */ exports.name = function () { console.log("My name is andyyou.") } ... /* main.js */ var person = require("./person.js") person.name()
/* person.js */ module.exports = "Hey, andyyou" // module.exports 不再是預設物件 exports.name = function () { // 透過 exports 捷徑掛進預設物件的內容, 將無法被揭露 console.log("My name is andyyou") } /* main.js */ var person = require("./person.js") // exports 的屬性被忽略了 person.name() // TypeError: Object Hey, andyyou has no method "name"
exports 只是指向 module.exports 的參考(Reference)
module.exports 初始值為 {} 空物件,於是 exports 也會取得該空物件
require() 回傳的是 module.exports 而不是 exports
所以您可以使用 exports.property_name = something 而不會使用 exports = something
一旦使用 exports = something 參考關係便會停止,也就是 exports 的資料都會被忽略。
本質上我們可以理解為所有模組都隱含實作了下面這行程式碼 (實際上 module 與 exports 是由 node 模組系統傳入給我們的模組的)
var exports = module.exports = {}
現在我們知道了,當我們要匯出一個 function 時我們得使用 module.exports。
如果令捷徑 exports 變數指向該 function,僅僅是捷徑 exports 對 module.exports 的關係被打斷而已,真正揭露的還是 module.exports。
另外,我們在許多專案看到下面的這行程式碼
exports = module.exports = something
這行程式碼作用就是確保 exports 在 module.exports 被我們覆寫之後,仍可以指向相同的參考。
接著我們就可以透過 module.exports 來定義並匯出一個 function
/* function.js */ module.exports = function () { return { name: "andyyou" } }
使用的方式則是
var func = require("./function")
關於 require 一個很重要的行為就是它會快取(Cache) module.exports 的值,未來每一次 require 被調用時都會回傳相同的值。
它會根據匯入檔案的絕對路徑來快取,所以當我們想要模組能夠回傳不同得值時,我們就需要匯出 function,如此一來每次執行函式時就會回傳一個新值。
下面在 Node REPL 中簡易的示範
$ node > f1 = require("/Users/andyyou/Projects/export_this/function") [Function] > f2 = require("./function") // 相同路徑 [Function] > f1 === f2 true > f1() === f2() false
您可以觀察到 require 回傳了同樣的函式物件實例,但每一次調用函式回傳的物件是不同的。
更詳細的介紹可以參考官方文件,值得一讀。
現在,我們可以開始探討介面的設計模式(pattern)了。
匯出命名空間一個簡單且常用的設計模式就是匯出一個包含數個屬性的物件,這些屬性具體的內容主要是函式,但並不限於函式。
如此,我們就能夠透過匯入該模組來取得這個命名空間下一系列相關的功能。
當您匯入一個命名空間類型的模組時,我們通常會將模組指定到某一個變數,然後透過它的成員(物件屬性)來存取使用這些功能。
甚至我們也可以將這些變數成員直接指定到區域變數。
var fs = require("fs") var readFile = fs.readFile var ReadStream = fs.ReadStream readFile("./file.txt", function (err, data) { console.log("readFile contents: %s", data) })
這便是fs 核心模組的做法
var fs = exports
首先用一個新的區域變數 fs,令其為捷徑 exports,因此預設情況下 fs 就指向了 module.exports 身上的預設物件。上面這一行我們可以說,只是將捷徑變數換個名稱成為 fs 如此而已。
接下來,我們就可以使用新的捷徑 fs 了,例如: fs.Stats = binding.Stats。
fs.readFile = function (path, options, callback_) { // ... }
其他東西也是一樣的作法,例如匯出建構子
fs.ReadStream = ReadStream function ReadStream(path, options) { // ... } ReadStream.prototype.open = function () { // ... }
當匯出命名空間時,您可以指定屬性到 exports ,就像 fs 的作法,又或者可以建立一個新的物件指派給 module.exports
/* exports 作法 */ exports.verstion = "1.0" /* 或者 module.exports 作法 */ module.exports = { version: "1.0", doYourTasks: function () { // ... } }
一個常見的作法就是透過一個根模組(root)來彙整並匯出其他模組,如此一來只需要一個 require 便可以使用所有的模組。
原文作者在Good Eggs工作時,會將資料模型(Model)拆分成個別的模組,並使用匯出建構子的方式匯出(請參考下文介紹),然後透過一個 index 檔案 來集合該目錄下所有的資料模型並一起匯出,如此一來在 models 命名空間下的所有資料模型都可以使用
var models = require("./models") var User = models.User var Product = models.Product
在 ES2015 和 CoffeeScript 中我們甚至還可以使用解構指派來匯入我們需要的功能
/* CoffeeScript */ {User, Product} = require "./models" /* ES2015 */ import {User, Product} from "./models"
而剛剛提到的 index.js 大概就如下
exports.User = require("./User") exports.Person = require("./person")
實際上這樣分開的寫法還有更精簡的寫法,我們可以透過一個小小的函式庫來匯入在同一階層中所有檔案並搭配 CamelCase 的命名規則匯出。
於是在我們的 index.js 中看起來就會如下
module.exports = require("../lib/require_siblings")(__filename)匯出函式
另外一個設計模式是匯出函式當作該模組的介面。常見的作法是匯出一個工廠函式(Factory function),然後呼叫並回傳一個物件。
在使用 Express.js 的時候便是這種作法
var express = require("express") var app = express() // 實際上 express() 傳回的 app 是一個 function,在 JS 中函式也是物件 app.get("/hello", function (req, res, next) { res.send("Hi there! We are using Express v" + express.version) })
Express 匯出該函式,讓我們可以用來建立一個新的 express 應用程式。
在使用這種模式時,通常我們會使用 factory function 搭配參數讓我們可以設定並回傳初始化後的物件。
想要匯出 function,我們就一定要使用 module.exports ,Express 便是這麼做
exports = module.exports = createApplication ... function createApplication () { ... }
上面指派了 createApplication 函式到 module.exports 然後再指給 exports 確保參考一致。
同時 Express 也使用下面這種方式將導出函式當作命名空間的作法使用。
exports.version = "3.1.1"
這邊要大略解釋一下由於 Javascript 原生並沒有支援命名空間的機制,於是大部分在 JS 中提到的 namespace 指的就是透過物件封裝的方式來達到 namespace 的效果,也就是第一種設計模式。
注意!並沒有任何方式可以阻止我們將匯出的函式作為命名空間物件使用,我們可以用其來引用其他的 function,建構子,物件。
Express 3.3.2 / 2013-07-03 之後已經將 exports.version 移除了
另外在匯出函式的時候最好為其命名,如此一來當出錯的時候我們比較容易從錯誤堆疊資訊中找到問題點。
下面是兩個簡單的例子:
/* bomb1.js */ module.exports = function () { throw new Error("boom") }
module.exports = function bomb() { throw new Error("boom") }
$ node > bomb = require("./bomb1"); [Function] > bomb() Error: boom at module.exports (/Users/andyyou/Projects/export_this/bomb1.js:2:9) at repl:1:2 ... > bomb = require("./bomb2"); [Function: bomb] > bomb() Error: boom at bomb (/Users/andyyou/Projects/export_this/bomb2.js:2:9) at repl:1:2 ...
匯出函式還有些比較特別的案例,值得用另外的名稱以區分它們的不同。
匯出高階函式一個高階函式或 functor 基本上就是一個函式可以接受一個或多個函式為其輸入或輸出。而這邊我們要談論的後者 - 一個函式回傳函式
當我們想要模組能夠根據輸入控制回傳函式的行為時,匯出一個高階函式就是一種非常實用的設計模式。
補充:functor & monad
舉例來說 Connect 就提供了許多可掛載的功能給網頁框架。
這裡的 middleware 我們先理解成一個有三個參數 (req, res, next) 的 function。
Express 從 v4.x 版之後不再相依於 connect
connect middleware 慣例就是匯出的 function 執行後,要回傳一個 middleware function。
在處理 request 的過程中這個回傳的 middleware function 就可以接手使用剛剛提到的三個參數,用來在過程中做一些處理或設定。
同時因為閉包的特性這些設定在整個中介軟體的處理流程中都是有效的。
舉例來說,compression 這個 middleware 就可以在處理 responsive 過程中協助壓縮
var connect = require("connect") var app = connect() // gzip outgoing responses var compression = require("compression") app.use(compression())
而它的原始碼看起來就如下
module.exports = compression ... function compression (options) { ... return function compression (req, res, next) { ... next() } }
於是每一個 request 都會經過 compression middleware 處理,而代入的 options 也因為閉包的關係會被保留下來
這是一種極具彈性的模組作法,也可能在您的開發項目上幫上許多忙。
匯出建構子middleware 在這裡您可以大略想成串連執行一系列的 function,自然其 Function Signature 要一致
在一般物件導向語言中,constructor 建構子指的是協助我們從類別 Class建立一個該類別物件實例的一段程式碼。
// C# class Car { // c# 建構子 // constructor 即 class 中用來初始化物件的 method。 public Car(name) { name = name; } } var car = new Car("BMW");
由於在 ES2015 之前 Javascript 並不支援類別,某種程度上在 Javascript 之中我們可以把任何一個 function 當作類別,或者說一個 function 可以當作 function 執行或者搭配 new 關鍵字當作 constructor 來使用。如果想知道更詳細的介紹可以閱讀MDN 教學。
欲匯出建構子,我們需要透過構造函式來定義類別,然後透過 new 來建立物件實例。
function Person (name) { this.name = name } Person.prototype.greet = function () { return "Hi, I am " + this.name } var person = new Person("andyyou") console.log(person.greet()) // Hi, I am andyyou
在這種設計模式底下,我們通常會將每個檔案設計成一個類別,然後匯出建構子。這使得我們的專案架構更加清楚。
var Person = require("./person") var person = new Person()
整個檔案看起來會如下
/* person.js */ function Person(name) { this.name = name } Person.prototype.greet = function () { return "Hi, I am " + this.name } exports = module.exports = Person匯出單一物件實例 Signleton
當我們需要所有的模組使用者共享物件的狀態與行為時,就需要匯出單一物件實例。
Mongoose是一個 ODM(Object-Document Mapper)函式庫,讓我們可以使用程式中的 Model 物件去操作 MongoDB。
var mongoose = require("mongoose") mongoose.connect("mongodb://localhost/test") var Cat = mongoose.model("Cat", {name: String}) var kitty = new Cat({name: "Zildjian"}) kitty.save(function (err) { if (err) throw Error("save failed") console.log("meow") })
那我們 require 取得的 mongoose 物件是什麼東西呢?事實上 mongoose 模組的內部是這麼處理的
function Mongoose() { ... } module.exports = exports = new Mongoose()
因為 require 的快取了 module.exports 的值,於是所有 reqire("mongoose") 將會回傳相同的物件實例,之後在整個應用程式之中使用的都會是同一個物件。
Mongoose 使用物件導向的設計模式來封裝,解耦(分離功能之間的相依性),維護狀態使整體具備可讀性,同時透過匯出一個 Mongoose Class 的物件給使用者,讓我們可以簡單的存取使用。
如果我們有需要,它也可以建立其他的物件實例來作為命名空間使用。實際上 Mongoose 內部提供了存取建構子的方法
Mongoose.prototype.Mongoose = Mongoose
因此我們可以這麼做
var mongoose = require("mongoose") var Mongoose = mongoose.Mongoose var anotherMongoose = new Mongoose() anotherMongoose.connect("mongodb://localhost/test")擴展全域物件
一個被匯入的模組不只限於單純取得其匯出的資料。它也可以用來修改全域物件或回傳全域物件,自然也能定義新的全域物件。而在這邊的全域物件(Global objects)或稱為標準內建物件像是 Object, Function, Array 指的是在全域能存取到的物件們,而不是當 Javascript 開始執行時所產生代表 global scope 的 global object。
當我們需要擴增或修改全域物件預設行為時就需要使用這種設計模式。當然這樣的方式是有爭議,您必須謹慎使用,特別是在開放原始碼的專案上。
例如:Should.js是一個常被用在單元測試中用來判斷分析 值 是否正確的函式庫。
require("should") var user = { name: "andyyou" } user.name.should.equal("andyyou")
這樣您是否比較清楚了,should.js 增加了底層的 Object 的功能,加入了一個非列舉型的屬性 should ,讓我們可以用簡潔的語法來撰寫單元測試。
而在內部 should.js 做了這樣的事情
var should = function (obj) { return new Assertion(util.isWrapperType(obj) ? obj.valueOf(): obj) } ... exports = module.exports = should Object.defineProperty(Object.prototype, "should", { set: function(){}, get: function(){ return should(this); }, configurable: true });
就算看到這邊您肯定跟我一樣有滿滿的疑惑,全域物件擴展定義跟 exports 有啥關聯呢?
事實上
/* whoami.js */ exports = module.exports = { name: "andyyou" } Object.defineProperty(Object.prototype, "whoami", { set: function () {}, get: function () { return "I am " + this.name } }) /* app.js */ var whoami = require("whoami") console.log(whoami) // { name: "andyyou" } var obj = { name: "lena" } console.log(obj.whoami) // I am lena
現在我們明白了上面說的修改全域物件的意思了。should.js 匯出了一個 should 函式但是它主要的使用方式則是把 should 加到 Object 屬性上,透過物件本身來呼叫。
套用猴子補丁(Monkey Patch)在這邊所謂的猴子補丁特別指的是在執行時期動態修改一個類別、模組或物件(也稱 object augmentation),通常會這麼做是希望補強某的第三方套件的 bug 或功能。
假設某個模組沒有提供您客製化功能的介面,而您又需要這個功能的時候,我們就會實作一個模組來補強既有的模組。
這個設計模式有點類似擴展全域物件,但並非修改全域物件,而是依靠 Node 模組系統的快取機制,當其他程式碼匯入該模組時去補強該模組的實例物件。
預設來說 Mongoose 會使用小寫以及複數的慣例替資料模型命名。例如一個資料模型叫做 CreditCard 最終我們會得到 collection 的名稱是 creditcards 。假如我們希望可以換成 credit_cards 並且其他地方也遵循一樣的用法。
下面是我們試著使用猴子補丁的方式來替既有的模組增加功能
var pluralize = require("pluralize") // 處理複數單字的函式庫 var mongoose = require("mongoose") var Mongoose = mongoose.Mongoose mongoose.Promise = global.Promise // v4.1+ http://mongoosejs.com/docs/promises.html var model = Mongoose.prototype.model // 補丁 var fn = function(name, schema, collection, skipInit) { collection = collection || pluralize.plural(name.replace(/([a-zd])([A-Z])/g, "$1_$2").toLowerCase()) return model.call(this, name, schema, collection, skipInit) } Mongoose.prototype.model = fn /* 實際測試 */ mongoose.connect("mongodb://localhost/test") var CreditCardSchema = new mongoose.Schema({number: String}) var CreditCardModel = mongoose.model("CreditCard", CreditCardSchema); var card = new CreditCardModel({number: "5555444433332222"}); card.save(function (err) { if (err) { console.log(err) } console.log("success") })
您不該輕易使用上面這種方式補丁,這邊只是為了說明猴子補丁這種方式,mongoose 已經有提供官方的方式設定名稱
var schema = new Schema({..}, { collection: "your_collection_name" })
當這個模組第一次被匯入的時候便會讓 mongoose 重新定義 Mongoose.prototype.model 並將其設回原本的 model 的實作。
如此一來所有 Mongoose 的實例物件都具備新的行為了。注意到這邊並沒有修改 exports 所以當我們 require 的時候得到的是預設的物件
另外如果您想使用上面這種補丁的方式時,記得閱讀原始碼並注意是否產生衝突。
請善用匯出的功能Node模組系統提供了一個簡單的機制來封裝功能,使我們能夠建立了清楚的介面。希望掌握這七種設計模式提供不同的優缺點能對您有所幫助。
在這邊作者並沒有徹底的調查所有的方式,一定有其他選項可供選擇,這邊只有描述幾個最常見且不錯的方法。
小結
namespace: 匯出一個物件包含需要的功能
root module 的方式,使用一個根模組匯出其他模組
function: 直接將 module.exports 設為 function
Function 物件也可以拿來當作命名空間使用
為其命名方便偵錯
exports = module.exports = something 的作法是為了確保參考(Reference)一致
high-order function: 可以透過代入參數控制並回傳 function 。
可協助實作 middleware 的設計模式
換句話說 middleware 即一系列相同 signature 的 function 串連。一個接一個執行
constructor: 匯出類別(function),使用時再 new,具備 OOP 的優點
singleton: 匯出單一物件實例,重點在各個檔案可以共享物件狀態
global objects: 在全域物件作的修改也會一起被匯出
monkey patch: 執行時期,利用 Node 快取機制在 instance 加上補丁
筆記一個 javascript 檔案可視為一個模組
解決特定問題或需求,功能完整由單一或多個模組組合而成的整體稱為套件(package)
require 匯入的模組具有自己的 scope
exports 只是 module.exports 的參考,exports 會記錄收集屬性如果 module.exports 沒有任何屬性就把其資料交給 module.exports ,但如果 module.exports 已經具備屬性的話,那麼exports 的所有資料都會被忽略。
就算 exports 置於後方仍會被忽略
Node 初始化的順序
Native Module -> Module
StartNodeInstance() -> CreateEnvironment() -> LoadEnvironment() -> Cached
Native Module 載入機制
檢查是否有快取
-> 有; 直接回傳 this.exports
-> 沒有; new 一個模組物件
cache()
compile() -> NativeModule.wrap() 將原始碼包進 function 字串 -> runInThisContext() 建立函式
return NativeModule.exports
Node 的 require 會 cache ,也就是說:如果希望模組產生不同的 instance 時應使用 function
資源官方 module 文件
理解 module exports
Export This: Interface Design Patterns for Node.js Modules
module.exports v.s exports
從 node.js 原始碼看 exports 與 module.exports
Export This 中文
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/80417.html
摘要:不過到底是怎麼保留的另外為什麼一個閉包可以一直使用區域變數,即便這些變數在該內已經不存在了為了解開閉包的神秘面紗,我們將要假裝沒有閉包這東西而且也不能夠用嵌套來重新實作閉包。 原文出處: 連結 話說網路上有很多文章在探討閉包(Closures)時大多都是簡單的帶過。大多的都將閉包的定義濃縮成一句簡單的解釋,那就是一個閉包是一個函數能夠保留其建立時的執行環境。不過到底是怎麼保留的? 另外...
摘要:的架構設計促使第三方開發者讓核心發揮出無限的潛力。當然建置比起開發是較進階的議題,因為我們必須要理解內部的一些事件。這個編譯結果包含的訊息包含模組的狀態,編譯後的資源檔,發生異動的檔案,被觀察的相依套件等。 本文將對 webpack 周邊的 middleware 與 plugin 套件等作些介紹,若您對於 webpack 還不了解可以參考這篇彙整的翻譯。 webpack dev ser...
摘要:不過這個效果感覺上就像是閃一下就切換到該位置。為了使用體驗上的感覺有時候網站會設計一種平滑捲動到該位置的效果。的方式非常簡單,只要在該元素設定注意是而不是這個方式非常方便不過目前只有支援,查閱。 眾所皆知 HTML 錨點(anchor link)透過給定標籤 id 屬性跳到頁面上特定位置的功能。不過這個效果感覺上就像是閃一下就切換到該位置。為了使用體驗上的感覺有時候網站會設計一種平滑捲...
摘要:不過這個效果感覺上就像是閃一下就切換到該位置。為了使用體驗上的感覺有時候網站會設計一種平滑捲動到該位置的效果。的方式非常簡單,只要在該元素設定注意是而不是這個方式非常方便不過目前只有支援,查閱。 眾所皆知 HTML 錨點(anchor link)透過給定標籤 id 屬性跳到頁面上特定位置的功能。不過這個效果感覺上就像是閃一下就切換到該位置。為了使用體驗上的感覺有時候網站會設計一種平滑捲...
摘要:目錄許多開發者會把的目錄命名為但這並不強迫。所有的檔案都會使用從被編譯成。同時有個小小的重點那就是我們可已觀察編譯後的檔案大小。在專案目錄下執行可以觀察截至目前為止的結果。我們的目標是要把編譯封裝到我們的中。 在今時今日,webpack 已經成為前端開發非常重要的工具之一。本質上它是一個 Javascript 模組封裝工具,但透過 loaders 和 plugins 它也可以轉換封裝其...
阅读 711·2021-11-22 13:52
阅读 1516·2021-09-27 13:36
阅读 2817·2021-09-24 09:47
阅读 2170·2021-09-22 15:48
阅读 3598·2021-09-22 15:39
阅读 1459·2019-08-30 12:43
阅读 2916·2019-08-29 18:39
阅读 3181·2019-08-29 12:51