一 .定義
定義一個操作中的算法框架,而將一些步驟延遲到子類中,使得子類可以不改變一個算法的結(jié)構(gòu)即可重定義該算法的某些特定步驟.
在面向?qū)ο箝_發(fā)過程中,通常會遇到這樣的一個問題,我們知道一個算法所需要的關(guān)鍵步驟,并確定了這些步驟的執(zhí)行順序,但是,某些步驟的具體實(shí)現(xiàn)是未知的,或者說某些步驟的實(shí)現(xiàn)會隨著環(huán)境的變化而改變,例如執(zhí)行程序的流程大致如下:
1.檢查代碼的正確性;
2.鏈接相關(guān)的類庫;
3.編譯相關(guān)代碼;
4.執(zhí)行程序.
對于不同程序設(shè)計語言,上述4個步驟都是不一樣的,但是,它們的執(zhí)行流程是固定的,這類問題的解決方案就是模板方法模式.
二.使用場景
1.多個子類有公有的方法,并且邏輯基本相同時;
2.重要,復(fù)雜的算法,可以把核心算法設(shè)計為模板方法,周邊的相關(guān)細(xì)節(jié)功能則由各個子類實(shí)現(xiàn);
3.重構(gòu)時,模板方法模式是一個經(jīng)常使用的模式,把相同的代碼抽取到父類中,然后通過鉤子函數(shù)約束其行為.
簡單實(shí)現(xiàn)
一樣是開計算機(jī)的流程,不同的人不同的打開密碼
public abstract class AbsComputer {
/*
* 設(shè)計模式 將相同的總結(jié)在一起 不相同的有子類完善
* */
private static final String TAG = "AbsComputer";
public void startUp(){
Log.d(TAG, "準(zhǔn)備啟動: ");
poweron();//按下電源
checkHardWare();//檢測硬件
loadOs(); //加載操作系統(tǒng)
login();
Log.d(TAG, "啟動成功: ");
}
protected abstract void login();//輸入密碼
protected void loadOs(){
Log.d(TAG, "加載操作系統(tǒng): ");
};
protected void checkHardWare(){
Log.d(TAG, "檢測硬件: ");
};
protected void poweron(){
Log.d(TAG, "按下電源: ");
}
}
軍隊(duì)
public class ArmyComputer extends AbsComputer {
private static final String TAG = "軍隊(duì)";
@Override
protected void checkHardWare() {
super.checkHardWare();
Log.d(TAG, "檢測防火墻: ");
}
@Override
protected void login() {
Log.d(TAG, "輸入了多重1024位密碼: ");
}
}
程序員
public class NbCoder extends AbsComputer {
private static final String TAG = "程序員";
@Override
protected void login() {
Log.d(TAG, "簡單的鎖 ");
}
}
小白
public class XiaoBai extends AbsComputer {
private static final String TAG = "小白";
@Override
protected void login() {
Log.d(TAG, "小白電腦沒有密碼,直接進(jìn)入操作頁面: ");
}
}
實(shí)現(xiàn)
private void template() {
new XiaoBai().startUp();
new NbCoder().startUp();
new ArmyComputer().startUp();
}
四.總結(jié)
模板方法模式用4個字概括:流程封裝;也就是把某個固定流程封裝到final函數(shù)中,并讓子類定制這個流程中某個或者某些步驟;
優(yōu)點(diǎn):
1.封裝不變部分,擴(kuò)展可變部分;
2.提取公共部分代碼,便于維護(hù);
缺點(diǎn):
模板方法會帶來代碼閱讀的難度,會讓用戶覺得難以理解.