模板方法模式


模板方法模式 

  • 模板方法模式定義一個算法中的操作框架,而將一些步驟延遲到子類中。使得子類可以不改變算法的結構即可重定義該算法的某些特定步驟。(Define the skeleton of an algorithm in an operation, deferring some steps to subclasses. Template Method lets subclasses redefine certain steps of an algorithm without changing the algorithm's structure)

  • 模板方法模式的通用類圖非常簡單,僅僅使用了Java的繼承機制,但它是一個非常廣泛的模式。其類圖如下,其中AbstractClass叫做抽象模板,它的方法分為兩類:
    • 基本方法:是由子類實現的方法,並且在模板方法被調用。(一般都加上final關鍵字,防止被覆寫)
    • 模板方法:可以有一個或幾個,一般是一個具體方法,也就是一個框架,實現對基本方法的調用,完成固定的邏輯。(抽象模板中的基本方法盡量設計為protected類型,符合迪米特法則,不需要暴露的屬性或方法盡量不要設置為protected類型。實現類若非必要,盡量不要擴大父類中的訪問權限)

 

    • 上述通用類圖的源碼如下:
    • public abstract class AbstractClass {
          protected abstract void doAnything();
          protected abstract void doSomething();
          public final void templateMethod(){
              /*
               * 調用基本方法,完成相關的邏輯
               */
              this.doAnything();
              this.doSomething();
          }
      }
      public class ConcreteClass1 extends AbstractClass {
      
          @Override
          protected void doAnything() {
              // TODO Auto-generated method stub
              //子類實現具體
          }
      
          @Override
          protected void doSomething() {
              // TODO Auto-generated method stub    
          }
      }
      public class ConcreteClass2 extends AbstractClass {
      
          @Override
          protected void doAnything() {
              // TODO Auto-generated method stub
              //子類實現具體
          }
      
          @Override
          protected void doSomething() {
              // TODO Auto-generated method stub    
          }
      }

       

  • 模板方法模式的優點

    • 封裝不變部分,擴展可變部分。把認為不變部分的算法封裝到父類中實現,而可變部分的則可以通過繼承來繼續擴展。
    • 提取公共部分代碼,便於維護。
    • 行為由父類控制,子類實現。
  • 模板方法模式的缺點

    • 按照設計習慣,抽象類負責聲明最抽象、最一般的事物屬性和方法,實現類負責完成具體的事務屬性和方法,但是模板方式正好相反,子類執行的結果影響了父類的結果,會增加代碼閱讀的難度。
  • 模板方法模式的使用場景

    • 多個子類有共有的方法,並且邏輯基本相同
    • 重要、復雜的算法,可以把核心算法設計為模板方法,周邊的相關細節功能則由各個子類實現
    • 重構時,模板方法是一個經常使用的方法,把相同的代碼抽取到父類中,然后通過構造函數約束其行為。
  • 模板方法模式的擴展

如上通用類圖中,若在ConcreteClass2中不想執行doSomething()方法,那么需要對模板方法做一些修改,其通用類圖如下:

 

    • 通用類圖的源碼如下
    • public abstract class AbstractClass {
          protected abstract void doAnything();
          protected abstract void doSomething();
          protected boolean isDoSomething(){ //父類方法返回真
              return true;
          }
          public final void templateMethod(){
              /*
               * 調用基本方法,完成相關的邏輯
               */
              this.doAnything();
              if(this.isDoSomething())
                  this.doSomething();
          }
      }
      public class ConcreteClass1 extends AbstractClass {
          private boolean isDoSth;
          @Override
          protected void doAnything() {
              // TODO Auto-generated method stub
              //子類實現具體
          }
      
          @Override
          protected void doSomething() {
              // TODO Auto-generated method stub    
          }
          protected void setDo(boolean isDo){
              this.isDoSth = isDo;
          }
          protected boolean isDoSomething(){
              return isDoSth;
          }
      }

       

  • 最佳實踐

模板方法模式是通過父類建立框架,子類在重寫了父類部分方法之后,在調用從父類繼承的方法,產生不同的效果,通過修改子類,影響父類行為的結果,模板方法在一些開源框架中應用非常多,它提供了一個抽象類,然后開源框架寫了一堆子類,如果需要擴展功能,可以繼承此抽象類,然后覆寫protected基本方法,然后在調用一個類似TemplateMethod()的模板方法,完成擴展開發。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM