设计模式学习--模板方法模式(Tamplate Pattern)
2013年6月27日 设计模式学习
回顾知识+新的模式
OO基础
OO原则
- 封装变化
- 多用组合,少用继承
- 针对接口编程,不针对实现编程
- 为交互对象之间的松耦合设计而努力
- 类应该对扩展开放,对修改关闭
- 依赖抽象,不要依赖具体类
- 只和朋友交谈
- 别找我,我会找你(这是新的原则:由超类主控一切,当它们需要的时候,自然会去调用子类,这就跟好莱坞一样)
OO模式
模板模式——在一个方法中定义一个算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算结构的情况下,重新定义算法中的某些步骤。
模板方法模式要点
- “模板方法”定义 算法的步骤,把这些步骤的实现延迟到子类。
- 模板方法模式为我们提供一种代码复用的重要技巧。
- 模板方法的抽象类可以定义具体方法、抽象方法和钩子。
- 抽象方法由子类实现。
- 钩子是一种方法,它在抽象类中不做事,或者只做默认的事情,子类可以选择要不要去覆盖它。
- 为了防止子类改变模板方法中的算法,可以将模板方法声明为final。
- 好莱坞原则告诉我们,将决策权放在高层模块中,以便决定如何以及何时调用低层模块。
- 你将在真实世界代码中看到模块方法模式的许多变体,不要期待它们全都是一眼就可以被你认出。
- 策略模式和模板方法模式都封装算法,一个用组合,一个用继承。
- 工厂方法是模板方法的一种特殊版本。
模板方法模式使用例子:泡茶和泡咖啡的冲泡
来点咖啡因吧
这是一个声明为抽象的类,里面定义了一个模板方法,两个由子类去实现的抽象方法
package simpleTamplateMethod;
/**
* 2013/6/27
* @author wwj
* 模板方法模式
*/
public abstract class CaffineBeverage {
/**
* 模板方法
* 1.它是一个方法
* 2.它用作一个算法的模板,在这个例子中,算法是用来制作咖啡因饮料的
* 在这个模板中,算法内的每一个步骤都被一个方法代表了
*/
final void prepareRecipe() {
boilWater();
brew();
pourInCup();
addCondiments();
}
//需要子类提供的方法,必须要在超类中声明为抽象
abstract void brew();
abstract void addCondiments();
void boilWater() {
System.out.println("Boiling water");
}
void pourInCup() {
System.out.println("Pouring into cup");
}
Coffee和Tea两个类继承CaffineBeverage超类
package simpleTamplateMethod;
/**
* 咖啡继承咖啡因饮料
* @author wwj
*
*/
public class Coffee extends CaffineBeverage {
@Override
void brew() {
System.out.println("Dripping Coffee through filter"); //冲泡咖啡
}
@Override
void addCondiments() {
System.out.println("Adding Sugar and Milk"); //增加糖和牛奶
}
}
package simpleTamplateMethod;
/**
* 茶类继承自咖啡因饮料
* @author wwj
*
*/
public class Tea extends CaffineBeverage{
@Override
void brew() {
System.out.println("Steeping the tea"); //泡茶
}
@Override
void addCondiments() {
System.out.println("Adding Lemon"); //增加柠檬
}
}
测试一下
package simpleTamplateMethod;
public class CaffineBeverageTest {
public static void main(String[] args) {
System.out.println("-------泡茶咯----------");
Tea myTea = new Tea();
myTea.prepareRecipe(); //调用模板方法,就这样把所有工作做完了
System.out.println("-------泡茶咯----------");
System.out.println("-------煮咖啡咯----------");
Coffee myCoffee = new Coffee();
myCoffee.prepareRecipe();
System.out.println("-------煮咖啡咯----------");
}
}
测试结果
-------泡茶咯----------
Boiling water
Steeping the tea
Pouring into cup
Adding Lemon
-------泡茶咯----------
-------煮咖啡咯----------
Boiling water
Dripping Coffee through filter
Pouring into cup
Adding Sugar and Milk
-------煮咖啡咯----------
对模板方法进行挂钩
package hook;
/**
* 模块方法使用钩子
* @author wwj
*
*/
public abstract class CaffineBeverageWithHook {
void prepareRecipe() {
boilWater();
brew();
pourInCup();
if(customerWantsCondiments()){ //加上这个条件,通过一个具体方法来决定是否执行语句,这就是对模块方法进行挂钩
addCondiments();
}
}
abstract void brew();
abstract void addCondiments();
void boilWater() {
System.out.println("Boilng water");
}
void pourInCup() {
System.out.println("Pouring into cup");
}
/**
* 这就是钩子
* @return
*/
boolean customerWantsCondiments() {
return true;
}
}
package hook;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
public class CoffeeWithHook extends CaffineBeverageWithHook {
@Override
void brew() {
System.out.println("Dripping Coffee through filter");
}
@Override
void addCondiments() {
System.out.println("Adding Sugar and Milk");
}
/**
* 覆盖这个钩子函数,提供自己的功能
*/
public boolean customerWantsCondiments() {
String answer = getUserInput();
if(answer.toLowerCase().startsWith("y")){
return true;
} else {
return false;
}
}
private String getUserInput() {
String answer = null;
System.out.println("Would you like milk and sugar with your coffee (y/n)? ");
BufferedReader in = new BufferedReader(new InputStreamReader(System.in));
try {
answer = in.readLine();
} catch (IOException e) {
System.out.println("IO error trying to read your answer");
}
if(answer == null) {
return "no";
}
return answer;
}
}
package hook;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
public class TeaWithHook extends CaffineBeverageWithHook {
@Override
void brew() {
System.out.println("Steeping the tea");
}
@Override
void addCondiments() {
System.out.println("Adding Lemon");
}
/**
* 覆盖这个钩子函数,提供自己的功能
*/
public boolean customerWantsCondiments() {
String answer = getUserInput();
if(answer.toLowerCase().startsWith("y")){
return true;
} else {
return false;
}
}
private String getUserInput() {
String answer = null;
System.out.println("Would you like lemon with your tea (y/n)? ");
BufferedReader in = new BufferedReader(new InputStreamReader(System.in));
try {
answer = in.readLine();
} catch (IOException e) {
System.out.println("IO error trying to read your answer");
}
if(answer == null) {
return "no";
}
return answer;
}
}
package hook;
public class BeverageTestDrive {
public static void main(String[] args) {
TeaWithHook teaHook = new TeaWithHook();
CoffeeWithHook coffeeWithHook = new CoffeeWithHook();
System.out.println("\nMaking tea...");
teaHook.prepareRecipe();
System.out.println("\nMaking coffee...");
coffeeWithHook.prepareRecipe();
}
}
测试结果:
Making tea...
Boilng water
Steeping the tea
Pouring into cup
Would you like lemon with your tea (y/n)?
y
Adding Lemon
Making coffee...
Boilng water
Dripping Coffee through filter
Pouring into cup
Would you like milk and sugar with your coffee (y/n)?
y
Adding Sugar and Milk