模板方法模式

定義

模板方法模式由兩部分構成,第一部分是抽象父類,第二部分是具體的實現子類。通常在抽象父類中封裝了子類的算法框架,包括實現一些公共方法以及封裝子類中所有方法的執行順序。子類的算法框架即是模板方法。

要點

  • 在模板方法模式中,子類實現中的相同部分被上移到父類中,而將不同的部分留待子類來實現。
  • 模板方法作為一個算法的模板,指導子類以何種順序去執行哪些方法。
  • 抽象類:模板方法模式是一種嚴重依賴抽象類的設計模式。
  • 在 JavaScript 中使用模板方法模式時,沒有辦法保證子類會重寫父類中的抽象方法。一種解決方案是在創建對象的時候,用鴨子類型來模擬接口檢查,缺點是增加了不必要的複雜性,在業務代碼中添加了跟業務邏輯無關的代碼;另一種解決方案是讓父類的抽象方法直接拋出一個異常,缺點是直到程序運行的時候才知道哪裡出了錯。
  • 鉤子方法:放置鉤子是隔離變化的一種常用手段。我們在父類中容易變化的地方放置鉤子,鉤子可以有一個默認的實現,究竟要不要「掛鉤」,根據子類鉤子方法的返回結果決定。
  • 好萊塢原則和模板方法模式:模板方法模式中,子類放棄了對自己的控制權,而是改為父類通知子類,哪些方法應該在什麼時候被調用。作為子類,只負責提供實現上的細節。
  • 在 JavaScript 中,我們可以通過高階函數的方式,而不是傳統的繼承的方式,來更好地實現一個模板方法模式。

核心代碼

var Beverage = function(param) {
    var boilWater = function() {
        \\ ...
    };
    var brew = param.brew || function() {
        throw new Error('必須傳遞 brew 方法');
    };
    var pourInCup = param.pourInCup || function() {
        throw new Error('必須傳遞 pourInCup 方法');
    };
    var addCondiments = param.addCondiments || function() {
        throw new Error('必須傳遞 addCondiments 方法');
    };

    var F = function() {};
    F.prototype.init = function() {
        boilWater();
        brew();
        pourInCup();
        addCondiments();
    };

    return F;
};

var Tea = Beverage({
    brew: function() { \\ ... },
    pourInCup: function() { \\ ... },
    addCondiments: function() { \\ ... },
});
var tea = new Tea();
tea.init();

你可能感兴趣的:(模板方法模式)