定義
模板方法模式在一個方法中定義類一個算法的骨架茸歧,而將一些步驟延遲到子類中。模板方法使得子類可以在不改變算法結(jié)構(gòu)的情況下,重新定義算法中的某些步驟次舌。
解釋
甲打開了一個app伊约,買了一張汽車票姚淆,乘坐汽車,到了泰山屡律。
乙打開了一個app腌逢,買了一張船票,乘船超埋,到了泰山搏讶。
丙打開了一個app,買了一張機票霍殴,乘坐飛機媒惕,到了泰山。
骨架:某人繁成,打開了一個app吓笙,買了xx,乘坐xx巾腕,到了泰山面睛。
子類:甲乙丙提供部分算法的具體實現(xiàn)。
類圖
image.png
偽代碼
public abstract class Travel{
public final void travel(){// 算法步驟
openApp();
buyTicket();
takeVehicle();
to();
}
pulbic final void openApp(){// 限定只能打開指定的app 子類不能覆蓋
System.out.println("open app");
}
public abstract void buyTicket();//子類實現(xiàn)
public abstract void takeVehicle();//子類實現(xiàn)
public final void to(){
System.out.println("to 泰山");
}
}
public class 甲 extends Travel{
public void buyTicket(){ // 子類實現(xiàn)
System.out.println("buy car tickey");
}
public void takeVehicle(){// 子類實現(xiàn)
System.out.println("take car);
}
}
public class 乙 extends Travel{
public void buyTicket(){ // 子類實現(xiàn)
System.out.println("buy ship tickey");
}
public void takeVehicle(){// 子類實現(xiàn)
System.out.println("take ship);
}
}
示例:
Travel travel = new 甲();
travel.travel();
travel = new 乙();
travel.travel();
描述
- Travel就是一個算法模板尊搬,定義了一些列的算法步驟叁鉴,其中部分步驟是抽象的,需要子類實現(xiàn)佛寿。整個Travel的算法結(jié)構(gòu)是固定的幌墓。
模板方法中的鉤子
鉤子是一種被聲明在抽象類中的方法,但只有空的或者默認的實現(xiàn)冀泻。鉤子的存在常侣,可以讓子類有能力對算法的不同點進行掛鉤。要不要掛鉤弹渔,有子類自行決定胳施。
鉤子其實也是算法步驟中的一個步驟,只是有些特殊肢专,用于控制作用舞肆。
鉤子偽代碼
public abstract class Travel{
public final void travel(){// 算法步驟
openApp();
if(isGoodWeather()){// 好天氣才出行
buyTicket();
takeVehicle();
to();
}
}
pulbic final void openApp(){// 限定只能打開指定的app 子類不能覆蓋
System.out.println("open app");
}
public abstract void buyTicket();//子類實現(xiàn)
public abstract void takeVehicle();//子類實現(xiàn)
public final void to(){
System.out.println("to 泰山");
}
public boolean isGoodWeather(){//鉤子方法,默認好天氣
return true;
}
}
public class 甲 extends Travel{
public void buyTicket(){ // 子類實現(xiàn)
System.out.println("buy car tickey");
}
public void takeVehicle(){// 子類實現(xiàn)
System.out.println("take car);
}
public void isGoodWeather(){
return false;
}
}
public class 乙 extends Travel{
public void buyTicket(){ // 子類實現(xiàn)
System.out.println("buy ship tickey");
}
public void takeVehicle(){// 子類實現(xiàn)
System.out.println("take ship);
}
}
示例:
Travel travel = new 甲();
travel.travel();// 甲的天氣不好博杖,不能出行
travel = new 乙();
travel.travel();// 乙的天氣好椿胯,能出行
- isGoodWeather()其實也是算法的一個步驟,只是超類提供類默認實現(xiàn)剃根,子類選擇性實現(xiàn)哩盲。
- 鉤子的作用是讓子類有機會對模板方法中的步驟作出反應(yīng),作出一部分決定。
哪些方法需要定義為抽象方法
- 當你的子類“必須”提供算法中某個方法或步驟的實現(xiàn)時种冬,就使用抽象方法镣丑,如果算法是可選的,就用鉤子娱两。
模板方法與策略模式的卻別
- 概括來說莺匠,策略模式,使用對象組合十兢,組合的類實現(xiàn)了整個算法趣竣;模板方法,使用繼承旱物,子類實現(xiàn)整個算法步驟中的一部分步驟細節(jié)遥缕。
- 模板方法定義了一個算法的大綱,由子類定義其中某些步驟的內(nèi)容宵呛,這么一來单匣,在算法中的個別步驟可以有不同的實現(xiàn)細節(jié),但是算法的結(jié)構(gòu)依然維持不變宝穗。
- 策略模式户秤,定義了一個算法家族,并讓這些算法可以互換逮矛。正因為每一個算法都被封裝起來了鸡号,所以客戶可以輕輕一地使用不同的算法。
總結(jié)
- 模板方法定義了算法的步驟须鼎,把這些步驟的實現(xiàn)延遲到子類鲸伴。
- 模板方法模式為我們提供了一種代碼復(fù)用的重要技巧。
- 模板方法的抽象類可以定義具體方法晋控、抽象方法和鉤子汞窗。
- 抽象方法由子類實現(xiàn)。
- 鉤子是一種方法赡译,它在抽象類中不做事仲吏,或者只做默認的事情,子類可以選擇要不要覆蓋它捶朵。
- 為了防止子類改變模板方法中的算法蜘矢,可以將模板方法聲明為final狂男。
- 好萊塢原則告訴我們综看,將決策權(quán)放在高層模塊中,以便決定如何以及何時調(diào)用低層模塊岖食。
- 策略模式和模板方法模式都封裝算法红碑,一個用組合,一個用繼承。
- 工廠方法是模板方法的一種特殊版本析珊。