一. 概念
模板方法模式(Template Method Pattern),又叫模板模式(Template Pattern),在一个抽象类公开定义了执行它的方法的模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。
简单说,模板方法模式定义一个操作中的算法的骨架,而将一些步骤延迟到子类中,使得子类可以不改变一个算法的结构,就可以重定义该算法的某些特定步骤。这种类型的设计模式属于行为型模式。
二. 原理类图
AbstractClass:抽象类,类中实现了模板方法,定义了算法的骨架,具体子类需要去实现其它的抽象方法operationr2, 3, 4
ConcreteClass:实现抽象方法operationr2, 3, 4,以完成算法中特点子类的步骤
三. 代码示例
以制作豆浆为例,制作豆浆的流程为选材——添加配料——浸泡——放到豆浆机打碎,通过添加不同的配料,可以制作出不同口味的豆浆。选材、浸泡和放到豆浆机打碎这几个步骤对于制作每种口味的豆浆都是一样的,因此可以使用模板方法模式。
类图
SoyaMilk抽象类,定义模板方法,使用final防止子类覆盖,选材、浸泡、打碎方法一致直接实现,addIngredients()不相同采用抽象方法,表示制作豆浆
在模板方法模式的父类中,我们可以定义一个方法,它默认不做任何事,子类可以视情况要不要覆盖它,该方法称为“钩子”。isAddIngredients(),钩子方法决定是否添加调料
public abstract class SoyaMilk {
//模板方法,做成final,防止子类覆盖
final void make(){
select();
if(isAddIngredients()){
addIngredients();
}
soak();
beat();
}
//选材
void select(){
System.out.println("第一步:选好新鲜的黄豆");
}
//添加不同配料
abstract void addIngredients();
//浸泡
void soak(){
System.out.println("第三步:黄豆和配料浸泡在一起");
}
//打碎
void beat(){
System.out.println("第四步:放入豆浆机打碎");
}
//钩子方法,决定是否添加配料
boolean isAddIngredients(){
return true;
}
}
RedBeanSoyaMilk类,继承SoyaMilk抽象类,具体实现addIngredients() ,表示制作红豆豆浆
public class RedBeanSoyaMilk extends SoyaMilk{
@Override
void addIngredients() {
System.out.println("加入上好红豆");
}
}
PeanutSoyaMilk类,继承SoyaMilk抽象类,具体实现addIngredients() ,表示制作花生豆浆
public class PeanutSoyaMilk extends SoyaMilk{
@Override
void addIngredients() {
System.out.println("加入上好花生");
}
}
PureSoyaMilk类,继承SoyaMilk抽象类,重写钩子方法,表示不添加调料制作纯豆浆
public class PureSoyaMilk extends SoyaMilk{
@Override
boolean isAddIngredients() {
return false;
}
@Override
void addIngredients() {
}
}
Client类,客户端
public class Client {
public static void main(String[] args) {
System.out.println("------------------制作红豆豆浆----------------------");
RedBeanSoyaMilk redBeanSoyaMilk = new RedBeanSoyaMilk();
redBeanSoyaMilk.make();
System.out.println("------------------制作花生豆浆----------------------");
PeanutSoyaMilk peanutSoyaMilk = new PeanutSoyaMilk();
peanutSoyaMilk.make();
System.out.println("------------------制作纯豆浆----------------------");
PureSoyaMilk pureSoyaMilk = new PureSoyaMilk();
pureSoyaMilk.make();
}
}
输出结果