模板方法
ps:在软件开发的过程中,如果相同的一段代码复制过两次,就需要对设计产生怀疑,架构师要明确的说明为什么相同的逻辑要出现两次或者多次。
定义
Define the skeleton of an algorithm in an operation, deferring some steps to subclasses. Template Method lets subclasses redefine certain steps of an algorithm without changing the algorithm’s structure.(定义一个操作中的算法的框架,而将一些步骤延迟到子类中。使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。)
通用类图:
其中,AbstractClass叫做抽象模板,它的方法分为两类:
- 基本方法:也叫做基本操作,是由子类实现的,并且在模板方法中调用
- 模板方法:可以有一个或者几个,一般是具体方法,也就是一个框架,实现对基本方法的调度,完成固定的逻辑
ps:为了防止被恶意的操作,一般模板方法都会加上final关键字,不允许被覆写。
通用代码:
public abstract class AbstractClass {
// 基本方法
protected void doSomething();
//基本方法
protected void doAnything();
//模板方法
public void templateMethod(){
//调用基本方法,完成相关逻辑
this.doAnything();
this.doSomethign();
}
}
抽象模板中的基本方法尽量设计为protected,符合迪米特法则,不需要暴露的属性或者方法尽量不要设置为protected类型。实现类若非必要,尽量不要扩大父类中的访问权限。
应用
优点
- 封装不可变部分,扩展可变部分
- 提取公共代码,便于维护
- 行为由父类控制,子类实现
缺点
抽象类定义了部分抽象方法,由子类实现,子类的执行结果影响了父类的结果,也就是子类对父类产生了影响,在项目中,会带来代码阅读的难度。
使用场景
- 多个子类有公有的方法,并且逻辑基本相同
- 重要,复杂的算法,可以把核心的算法设计为模板方法,周边的细节功能则由各个子类实现
有公有的方法,并且逻辑基本相同 - 重要,复杂的算法,可以把核心的算法设计为模板方法,周边的细节功能则由各个子类实现
- 重构时,模板方法模式是一个经常使用模式,把相同的代码抽取到父类中,然后通过钩子函数约束其行为