模板方法模式:在一个方法中定义一个算法的骨架,而将一些步骤延迟到了子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。 == 封装算法 ==
模板方法定义了一个算法的步骤,并允许子类为每一个或多个步骤提供实现。
具体有以下使用经验:
1 父类中的模板方法不可以被重写,拥有算法并且保护这个算法(声明为final)。
2 可以代码复用最大化。
3 抽象父类方法专注在代码本身,而由子类提供完整的实现。由框架控制如何做事情,而使用框架的人只需要指定框架算法中的一些细节。
4 hook 钩子。父类中的一个方法,用来控制是否执行某段算法。返回值为boolean。
5 好莱坞原则 父类对子类说“别调用我们,我们会调用你”。两个类之间切忌相互调用。原则:底层组件(例如子类)可以参与计算,但是高层组件(例如父类)何时以及如何调用底层组件,底层组件绝对不可以直接调用高层组件。
代码:
抽象的父类:
//一个抽象的父类
public abstract class CaffeinBeverage {
//这个是我们的模板方法,定义了一个算法模板:用来制作饮料
//模板方法定义了算法,也即这几个方法的实现过程
public final void prepareRecipe() { //recipe 食谱,配方 不能重写了
boilWater();
brew();
pourInCup();
if(condimentHook()){
addCondiments();
}
}//模板实际就是一个方法
//这两个方法需要子类来实现
abstract void addCondiments() ;
abstract void brew() ;
//冲水和倒入杯子 由父类实现
private void pourInCup() {
System.out.println("==倒入杯子==");
}
private void boilWater() {
System.out.println("==把水煮沸==");
}
public boolean condimentHook(){ //钩子,用来控制是否执行某段算法(一段代码)
return true; //开关
}
}
子类一:
import java.util.Scanner;
public class TeaHook extends CaffeinBeverage {
private Scanner scanner;
@Override
void addCondiments() {
System.out.println("==添加柠檬==");
}
@Override
void brew() {
System.out.println("==放入茶叶==");
}
public TeaHook() {
super();
}
@Override
public boolean condimentHook() {
System.out.println("是否添加柠檬?y or n");
scanner = new Scanner(System.in);
char input = scanner.nextLine().charAt(0);
if(input == 'y'){
return true;
}else {
return false;
}
}
}
子类二:
import java.util.Scanner;
public class CoffeeHook extends CaffeinBeverage{
private Scanner scanner;
@Override
void addCondiments() {
System.out.println("==添加牛奶和糖==");
}
@Override
void brew() {
System.out.println("==放入咖啡==");
}
@Override
public boolean condimentHook() {
System.out.println("是否添加牛奶和糖?y or n");
scanner = new Scanner(System.in);
char input = scanner.nextLine().charAt(0);
if(input == 'y'){
return true;
}else {
return false;
}
}
}
测试:
public class MainTest {
public static void main(String[] args) {
TeaHook teaHook = new TeaHook();
CoffeeHook coffeeHook = new CoffeeHook();
System.out.println("泡一杯茶:");
teaHook.prepareRecipe();
System.out.println("泡一杯咖啡:");
coffeeHook.prepareRecipe();
}
}