這次想談談模組(Module),是 JavaScript 的設計模式的其中的一個,一般而言,我們都會希望程式有架構,被重覆使用,這時可以將程式分成一個個的模組,會用兩篇文章討論這個主題,這一篇是講述 Module pattern,下一篇會講述 ES2015 module。
var user = (function () {
// private function
function add() {
const count = $('li').length;
$('#list').append(`<li>Thomas${count}</li>`);
}
function remove() {
const count = $('li').length;
const $li = $('#list').find('li:last');
$li.remove();
}
return {
//public function
addUser() {
add();
},
removeUser() {
remove();
},
};
})();
Module pattern 的寫法就像上面這樣,使用 IIFE( Immediately Invoked Function Expression)將程式邏輯封裝,在 IIFE 中定義私有數數或是方法,只有要公開的方法才會傳出,這樣就很像 OOP 中的封裝概念,模擬私有變數及公有變數,範例程式碼在這邊可以看到。
var user = (function () {
function addUser() {
const count = $('li').length;
$('#list').append(`<li>Thomas${count}</li>`);
}
function removeUser() {
const count = $('li').length;
const $li = $('#list').find('li:last');
$li.remove();
}
return {
addUser: addUser,
removeUser: removeUser,
};
})();
另外一種 Module pattern 的寫法,我們都是用這種寫法,程式碼的架構變得更清楚,範例程式碼在這邊可以看到。
每個模組都是全域變數,又覺得有點鬆散及零亂,所以需要另一種型式來組合,這時需要命名空間的協助。
JavaScript 是沒有命名空間的,但可以使用物件來模擬。
const window.NS = window.NS || {};
在 window 下建立屬性,當有 NS 這個屬性時,就使用原本的 NS 屬性,不然初始化一個空物件,我們習慣一個模組一個檔案,所以有多個模組時,每個模組會存取到同一個命名空間。這邊是以 NS 為例子,可以是任何的名稱,我們習慣是以目前的專案名稱的大寫英文來命名。
(function () {
function addUser() {
const count = $('li').length;
$('#list').append(`<li>Thomas${count}</li>`);
}
function removeUser() {
const count = $('li').length;
const $li = $('#list').find('li:last');
$li.remove();
}
window.NS = window.NS || {};
window.NS.user = {
addUser: addUser,
removeUser: removeUser,
};
})();
這樣的寫法我覺得是 module pattern 的變形,window 下只有一個定義的屬性,也就是我所定義的命名空間物件,之後將所有模組都設為這命名空間的屬性。範例程式在這裡
namespace NS {
function addUser(){
const count = $('li').length;
$('#list').append(`<li>Thomas${count}</li>`);
}
function removeUser(){
const count = $('li').length;
const $li = $('#list').find('li:last');
$li.remove();
}
export const user = {
addUser: addUser,
removeUser: removeUser
}
}
TypeScript 本身就有命名空間的關鍵字,與 Module pattern 的寫法很相似,差別在於最後是用 export 的方式,傳出所要公開的 function。
並且我們發現,上面這段 TypeScript 轉成的 JavaScript(如上圖所示),與上節的 Module + Namespace 的程式碼很相似,也是先宣告全堿變數 NS,接著判斷 NS 是否已存在,之後將定義的模組設為 NS 的屬性。
Module pattern 我個人覺得是很方便的模式,可以將煩雜的程式碼切割成一塊塊的模組,讓它們彼此之間合作。