定義:
定義一個操作中的算法的框架,而將一些步驟延遲到子類中。使得子類可以不改變一個算法的結構即可重新定義該算法的某些特定步驟。
聽起來好高端的樣子,我的理解:
1.父類聲明了若干個抽象方法(基本方法)和若干個具體方法(模板方法)
2.抽象方法是一個算法(過程)的步驟,在子類中實現
3.模板方法是一個算法(過程)的框架,在父類中已經約定好,實現對基本方法調用,完成固定的邏輯
4.一個算法(過程)的結構在父類中定義,具體的實現細節則在子類中實現
注:為了防止惡意操作,一般模板方法都加上final,禁止重寫
通用類圖:
事實上,模版方法是編程中一個經常用到的模式。先來看一個例子,某日,程序員A拿到一個任務:給定一個整數數組,把數組中的數由小到大排序,然后把排序之后的結果打印出來。經過分析之后,這個任務大體上可分為兩部分,排序和打印,打印功能好實現,排序就有點麻煩了。但是A有辦法,先把打印功能完成,排序功能另找人做。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
abstract class AbstractSort { /** * 將數組array由小到大排序 * @param array */ protected abstract void sort( int [] array); public void showSortResult( int [] array){ this .sort(array); System.out.print( "排序結果:" ); for ( int i = 0 ; i < array.length; i++){ System.out.printf( "%3s" , array[i]); } } } |
寫完后,A找到剛畢業入職不久的同事B說:有個任務,主要邏輯我已經寫好了,你把剩下的邏輯實現一下吧。于是把AbstractSort類給B,讓B寫實現。B拿過來一看,太簡單了,10分鐘搞定,代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
class ConcreteSort extends AbstractSort { @Override protected void sort( int [] array){ for ( int i= 0 ; i<array.length- 1 ; i++){ selectSort(array, i); } } private void selectSort( int [] array, int index) { int MinValue = 32767 ; // 最小值變量 int indexMin = 0 ; // 最小值索引變量 int Temp; // 暫存變量 for ( int i = index; i < array.length; i++) { if (array[i] < MinValue){ // 找到最小值 MinValue = array[i]; // 儲存最小值 indexMin = i; } } Temp = array[index]; // 交換兩數值 array[index] = array[indexMin]; array[indexMin] = Temp; } } |
寫好后交給A,A拿來一運行:
1
2
3
4
5
6
7
|
public class Client { public static int [] a = { 10 , 32 , 1 , 9 , 5 , 7 , 12 , 0 , 4 , 3 }; // 預設數據數組 public static void main(String[] args){ AbstractSort s = new ConcreteSort(); s.showSortResult(a); } } |
運行結果:
1
|
排序結果: 0 1 3 4 5 7 9 10 12 32 |
模版方法模式的結構
模版方法模式由一個抽象類和一個(或一組)實現類通過繼承結構組成,抽象類中的方法分為三種:
抽象方法:父類中只聲明但不加以實現,而是定義好規范,然后由它的子類去實現。
模版方法:由抽象類聲明并加以實現。一般來說,模版方法調用抽象方法來完成主要的邏輯功能,并且,模版方法大多會定義為final類型,指明主要的邏輯功能在子類中不能被重寫。
鉤子方法:由抽象類聲明并加以實現。但是子類可以去擴展,子類可以通過擴展鉤子方法來影響模版方法的邏輯。
抽象類的任務是搭建邏輯的框架,通常由經驗豐富的人員編寫,因為抽象類的好壞直接決定了程序是否穩定性。
實現類用來實現細節。抽象類中的模版方法正是通過實現類擴展的方法來完成業務邏輯。只要實現類中的擴展方法通過了單元測試,在模版方法正確的前提下,整體功能一般不會出現大的錯誤。
模板方法模式的優點:
1. 封裝不變部分,擴展可變部分
2. 提取公共部分代碼,便于維護
3. 行為由父類控制,子類實現
模板方法模式的適用場景:
1. 多個子類有公共方法,并且邏輯基本相同
2. 對復雜的算法,核心算法設計為模板方法,細節功能則由各個子類實現
3. 重構代碼
模板方法模式的擴展
- 基本方法由于不需要對外提供訪問,因此可以設計為protected類型
- 設計鉤子方法(Hook Method):對外部提供接口,可以影響模板方法內的具體執行順序
總結:
父類建立框架,子類在重寫了父類部分方法后,再調用從父類繼承的方法,產生不同的結果。