1.模板方法模式概念
模板模式(Template Pattern)中芳誓,一個(gè)抽象類公開定義了執(zhí)行它的方法的方式/模板榆鼠。它的子類可以按需要重寫方法實(shí)現(xiàn),但調(diào)用將以抽象類中定義的方式進(jìn)行荆虱。這種類型的設(shè)計(jì)模式屬于行為型模式拌阴。
2.模板方法模式作用
在一個(gè)方法中定義一個(gè)算法的骨架,而將一些步驟延遲到子類中實(shí)現(xiàn)的烁。模板方法使得子類在不改變算法結(jié)構(gòu)的情況下褐耳,重新定義算法中的某些步驟。
3.使用場(chǎng)景
在造房子的時(shí)候撮躁,地基、走線买雾、水管都一樣把曼,只有在建筑的后期才有加壁櫥加?xùn)艡诘炔町悺?/p>
4.優(yōu)點(diǎn)和缺點(diǎn)
優(yōu)點(diǎn)
1、封裝不變部分漓穿,擴(kuò)展可變部分嗤军。
2、提取公共代碼晃危,便于維護(hù)叙赚。
3、行為由父類控制僚饭,子類實(shí)現(xiàn)震叮。
缺點(diǎn)
每一個(gè)不同的實(shí)現(xiàn)都需要一個(gè)子類來實(shí)現(xiàn),導(dǎo)致類的個(gè)數(shù)增加鳍鸵,使得系統(tǒng)更加龐大苇瓣。
5.例子解析
例子一:不使用鉤子的模板方法模式
抽象方法:CaffeineBeverage
public abstract class CaffeineBeverage {
final void prepareRecipe() {
boilWater();
brew();
pourInCup();
addCondiments();
}
public abstract void addCondiments();
public abstract void brew();
private void boilWater() {
System.out.println("Boiling water");
}
private void pourInCup() {
System.out.println("Pouring into cup");
}
}
具體類一:
public class Coffee extends CaffeineBeverage {
@Override
public void addCondiments() {
System.out.println("Adding Sugar and Milk");
}
@Override
public void brew() {
System.out.println("Dripping coffee through filter");
}
}
具體類二:
public class Tea extends CaffeineBeverage {
@Override
public void addCondiments() {
System.out.println("Adding Lemon");
}
@Override
public void brew() {
System.out.println("Steeping the tea");
}
}
客戶端:
public class Test {
public static void main(String[] args) {
Tea tea = new Tea();
tea.prepareRecipe();
System.out.println("\n");
Coffee coffee = new Coffee();
coffee.prepareRecipe();
}
}
例子二:使用鉤子的模板方法例子
模板方法抽象類:
public abstract class CaffeineBeverageWithHook {
final void prepareRecipe() {
boilWater();
brew();
pourInCup();
if (customerWantsCondiments()) {
addCondiments();
}
}
public abstract void addCondiments();
public abstract void brew();
private void boilWater() {
System.out.println("Boiling water");
}
private void pourInCup() {
System.out.println("Pouring into cup");
}
//我們?cè)谶@里定義一個(gè)方法,(通常)是空的缺省實(shí)現(xiàn)偿乖。
//“空的缺省實(shí)現(xiàn)”這句話的意思是击罪,首先“缺省”的意思是“默認(rèn)”的意思哲嘲,所以這句話的意思是定義一個(gè)默認(rèn)不做任何事情的方法
//這就是一個(gè)鉤子,子類可以選擇覆蓋這個(gè)方法媳禁,也可以選擇不覆蓋(即默認(rèn))
public boolean customerWantsCondiments() {
return true;
}
}
具體類:
public class CoffeeWithHook extends CaffeineBeverageWithHook {
@Override
public void addCondiments() {
System.out.println("Adding Sugar and Milk");
}
@Override
public void brew() {
System.out.println("Dripping coffee through filter");
}
@Override
public boolean customerWantsCondiments() {
String answer = getUserInput();
if (answer.toLowerCase().equals("y")) {
return true;
} else
return false;
}
private String getUserInput() {
String answer = null;
System.out
.println("Would you like sugar and milk with your caffee(y/n)?");
BufferedReader bis = new BufferedReader(
new InputStreamReader(System.in));
try {
answer = bis.readLine();
} catch (Exception e) {
System.err.println("IO error ,try to read your answer");
}
if (answer == null||answer.equals("")) {
return "no";
}
return answer;
}
}
那么眠副,鉤子到底是什么作用?
鉤子可以讓子類選擇是否實(shí)現(xiàn)算法中的可選部分竣稽。
6.總結(jié)
這些例子都是《Head First Design Patterns》一書里的囱怕,它里面還有幾個(gè)例子讓我們更深刻的了解什么事鉤子。下面總結(jié)一下了解到的知識(shí)丧枪。
1.鉤子是一個(gè)方法光涂,他在抽象類中不做事,或者只做默認(rèn)的事情拧烦,子類可以選擇是否重寫這個(gè)鉤子方法忘闻。
2.為了防止子類修改覆蓋抽象類中的模板方法,可以將模板方法設(shè)置為final恋博。
3.策略模式和模板模式都封裝了算法齐佳,只是它們的實(shí)現(xiàn)方式不同,策略模式使用的是組合债沮,模板模式使用的是繼承炼吴。
4.工廠方法是模板方法的一個(gè)特殊版本。
7.源碼下載
http://download.csdn.net/detail/lgywsdy/9750972