一、模板方法模式介绍
模板方法模式又叫模板模式。在一个抽象类公开定义了它的方法模板,它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。简单的说,模板方法模式定义一个操作中的算法骨架,而将一些步骤延迟到子类中,使得子类可以不改变一个算法的结构,就可以重定义该算法的某些特定步骤。
二、模板方法模式原理图
模板方法模式的原理图如下所示:
说明:
1)abstractClass抽象类:类中实现了模板方法,定义了算法的骨架,具体子类需要去实现其他的抽象方法operation2,3,4
2)ConcreteClass实现抽象方法operation2,3,4,以完成算法中特点子类的步骤
三、模板方法模式例子
编写制作豆浆的程序,说明如下:1)制作豆浆的流程:选材 ---> 添加配料 ---> 浸泡 ---> 放入豆浆机打碎;2)通过添加不同的配料,可以制作不同口味的豆浆;3)选材、浸泡和放在豆浆机打碎这几个步骤对于制作每种口味的豆浆都是一样的。
类图如下所示:
代码如下所示:
创建SoyaMilk抽象类及其子类:
public abstract class SoyaMilk{
//模板方法,模板方法可以做成final,不让子类去覆盖
final void make(){
select();
addCondiments();
soak();
beat();
}
//选材料
void select(){
System.out.println("第一步:选择好的新鲜黄豆");
}
//添加不同的配料,抽象方法,子类具体实现
abstract void addCondiments();
//浸泡
void soak(){
System.out.println("第三步:黄豆和配料开始浸泡,需要3个小时");
}
//打碎
void beat(){
System.out.println("第四步:黄豆和配料放在豆浆机打碎");
}
}
public class BlackBeanSoyaMilk extends SoyaMilk{
void addCondiments(){
System.out.println("加入好的黑豆");
}
}
public class PeanutSoyaMilk extends SoyaMilk{
void addCondiments(){
System.out.println("加入好的花生");
}
}
创建client类:
public class Client{
public static void main(String[] args){
System.out.println("-------制作黑豆豆浆--------");
SoyaMilk blackSoyaMilk = new BlackBeanSoyaMilk();
blackSoyaMilk.make();
System.out.println("-------制作花生豆浆--------");
SoyaMilk peanutSoyaMilk = new PeanutSoyaMilk();
peanutSoyaMilk .make();
}
}
四、模板方法模式的钩子方法
在模板方法模式的父类中,我们可以定义一个方法,它默认不做任何事,子类可以视情况要不要覆盖它,该方法称为“钩子” 。还是用上面做豆浆的例子来讲解,比如我们还希望制作豆浆,不添加任何的配料,请使用钩子方法对前面的模板方法进行改造。
public abstract class SoyaMilk{
//模板方法,模板方法可以做成final,不让子类去覆盖
final void make(){
select();
if(customeWantCondiments){
addCondiments();
}
soak();
beat();
}
//选材料
void select(){
System.out.println("第一步:选择好的新鲜黄豆");
}
//添加不同的配料,抽象方法,子类具体实现
abstract void addCondiments();
//浸泡
void soak(){
System.out.println("第三步:黄豆和配料开始浸泡,需要3个小时");
}
//打碎
void beat(){
System.out.println("第四步:黄豆和配料放在豆浆机打碎");
}
//钩子方法,决定是否需要添加配料
boolean customeWantCondiments(){
return true;
}
}
public class BlackBeanSoyaMilk extends SoyaMilk{
void addCondiments(){
System.out.println("加入好的黑豆");
}
}
public class PeanutSoyaMilk extends SoyaMilk{
void addCondiments(){
System.out.println("加入好的花生");
}
}
public class PurseSoyaMilk extends SoyaMilk{
void addCondiments(){
//空实现
}
boolean customerWantCondiments(){
return false;
}
}
public class Client{
public static void main(String[] args){
System.out.println("-------制作黑豆豆浆--------");
SoyaMilk blackSoyaMilk = new BlackBeanSoyaMilk();
blackSoyaMilk.make();
System.out.println("-------制作花生豆浆--------");
SoyaMilk peanutSoyaMilk = new PeanutSoyaMilk();
peanutSoyaMilk .make();
System.out.println("-------制作纯豆浆--------");
SoyaMilk purseSoyaMilk = new PurseSoyaMilk();
purseSoyaMilk .make();
}
}
五、模板方法模式的注意事项和细节
1)基本思想是:算法只存在于一个地方,也就是在父类中,容易修改。需要修改算法时,只要修改父类的模板方法或者已经实现了某些步骤,子类就会继承这些修改
2)实现了最大化代码复用。父类的模板方法和已实现的某些步骤会被子类继承而直接使用
3)既统一了算法,也提供了很大的灵活性。父类的模板方法确保了算法的结构保持不变,同时由子类提供部分步骤的实现
4)该模式的不足之处:每一个不同的实现都需要一个子类实现,导致类的个数更加,使得系统更加庞大
5)一般模板方法都加上final关键字,防止子类重写模板方法
6)模板方法模式使用场景:当要完成在某个过程,该过程要执行一系列步骤,这一系列的步骤基本相同,但其个别步骤在实现时可能不同,通常考虑用模板方法模式来处理