程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> JAVA編程入門知識 >> 深入淺出Java模式設計之模板方法模式

深入淺出Java模式設計之模板方法模式

編輯:JAVA編程入門知識
  一、引子
  
  這是一個很簡單的模式,卻被非常廣泛的使用。之所以簡單是因為在這個模式中僅僅使用到了繼續關系。
  
   <!-- frame contents --> <!-- /frame contents -->   繼續關系由於自身的缺陷,被專家們扣上了“罪惡”的帽子。“使用委派關系代替繼續關系”,“盡量使用接口實現而不是抽象類繼續”等等專家警告,讓我們這些菜鳥對繼續“另眼相看”。
  
  其實,繼續還是有很多自身的優點所在。只是被大家濫用的似乎缺點更加明顯了。合理的利用繼續關系,還是能對你的系統設計起到很好的作用的。而模板方法模式就是其中的一個使用范例。
  
  二、定義與結構
  
  GOF給模板方法(Template Method)模式定義一個操作中的算法的骨架,而將一些步驟延遲到子類中。使得子類可以不改變一個算法的結構即可重定義該算法的某些特定步驟。這裡的算法的結構,可以理解為你根據需求設計出來的業務流程。特定的步驟就是指那些可能在內容上存在變數的環節。
  
  可以看出來,模板方法模式也是為了巧妙解決變化對系統帶來的影響而設計的。使用模板方法使系統擴展性增強,最小化了變化對系統的影響。這一點,在下面的舉例中可以很明顯的看出來。
  
  來看下這個簡單模式的結構吧:
  
  1) AbstractClass(抽象類):定義了一到多個的抽象方法,以供具體的子類來實現它們;而且還要實現一個模板方法,來定義一個算法的骨架。該模板方法不僅調用前面的抽象方法,也可以調用其他的操作,只要能完成自身的使命。
  
  2) ConcreteClass(具體類):實現父類中的抽象方法以完成算法中與特定子類相關的步驟。
  
  下面是模板方法模式的結構圖。直接把《設計模式》上的圖拿過來用下:
  
  
  三、舉例
  
  還是在我剛剛分析完源碼的JUnit中找個例子吧。JUnit中的TestCase以及它的子類就是一個模板方法模式的例子。在TestCase這個抽象類中將整個測試的流程設置好了,比如先執行Setup方法初始化測試前提,在運行測試方法,然後再TearDown來取消測試設置。但是你將在Setup、TearDown裡面作些什麼呢?鬼才知道呢!!因此,而這些步驟的具體實現都延遲到子類中去,也就是你實現的測試類中。
  
  來看下相關的源代碼吧。
  
  這是TestCase中,執行測試的模板方法。你可以看到,裡面正像前面定義中所說的那樣,它制定了“算法”的框架——先執行setUp方法來做下初始化,然後執行測試方法,最後執行tearDown釋放你得到的資源。
  
   public void runBare() throws Throwable {
   setUp();
  
   try {
  runTest();
   }
  
   finally {
  tearDown();
   }
  }
  這就是上面使用的兩個方法。與定義中不同的是,這兩個方法並沒有被實現為抽象方法,而是兩個空的無為方法(被稱為鉤子方法)。這是因為在測試中,我們並不是必須要讓測試程序使用這兩個方法來初始化和釋放資源的。假如是抽象方法,則子類們必須給它一個實現,不管用到用不到。這顯然是不合理的。使用鉤子方法,則你在需要的時候,可以在子類中重寫這些方法。
  
   protected void setUp() throws Exception {}
  protected void tearDown() throws Exception {}
  四、適用情況
  
  根據上面對定義的分析,以及例子的說明,可以看出模板方法適用於以下情況:
  
  1) 一次性實現一個算法的不變的部分,並將可變的行為留給子類來實現。
  
  2) 各子類中公共的行為應被提取出來並集中到一個公共父類中以避免代碼重復。其實這可以說是一種好的編碼習慣了。
  
  3) 控制子類擴展。模板方法只在特定點調用操作,這樣就只答應在這些點進行擴展。比如上面runBare()方法就只在runTest前面適用setUp方法。假如你不願子類來修改你的模板方法定義的框架,你可以采用兩種方式來做:一是在API中不體現出你的模板方法;二、將你的模板方法置為final就可以了。
  
  可以看出,使用模板方法模式可以將代碼的公共行為提取出來,達到復用的目的。而且,在模板方法模式中,是由父類的模板方法來控制子類中的具體實現。這樣你在實現子類的時候,根本不需要對業務流程有太多的了解。
 
  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved