2.4 模板方法模式
Template Method Pattern
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.
定义一个操作中的算法框架,将一些步骤延迟到子类中。子类不用改变算法的结构就能重新定义该算法的某些特定步骤。
优点:
- 封装不变部分,扩展可变部分
把认为是不变的部分封装到父类实现,可变部分通过继承来继续扩展
- 提起公共部分代码,便于维护
- 行为由父类控制,子类实现
基本方法是由子类实现的,因此子类可以通过扩展的方式增加相应的功能,符合开闭原则。
缺点:
- 模板方法模式中,抽象类不仅定义了抽象方法也定义了具体的方法,而且子类的实现结果会影响到父类,在复杂项目中会带来代码阅读难度
使用场景:
- 多个子类有共有的方法,并且逻辑基本相同
- 重要、复杂的算法,可以把核心算法设计为模板方法,周边的相关细节功能则由各个子类实现。
- 重构时,模板方法模式是一个经常使用的模式,吧相同的代码抽取到父类,然后通过钩子函数约束其行为。
代码:
/**
* ❉ ❉ ❉ ❉ ❉ 抽象模板 ❉ ❉ ❉ ❉ ❉
*
* 抽象模板的基本方法尽量设计为protected,符合迪米特法则
*/
abstract class AbstractClass{
//基本方法
protected abstract void doSomething();
//基本方法
protected abstract void doAnything();
//模板方法,注意:为了防止恶意的操作,一遍模板方法都加上final关键字,不允许被覆写。
public void templeteMethod(){
this.doSomething();
this.doAnything();
}
}
/**
* ❉ ❉ ❉ ❉ ❉ 具体模板 ❉ ❉ ❉ ❉ ❉
*/
class ConcreateClass1 extends AbstractClass{
protected void doSomething(){}
protected void doAnything(){}
}
class ConcreateClass2 extends AbstractClass{
protected void doSomething(){}
protected void doAnything(){}
}
/**
* ❉ ❉ ❉ ❉ ❉ 场景 ❉ ❉ ❉ ❉ ❉
*/
class Client{
public static void main(String[] args) {
ConcreateClass1 cls1 = new ConcreateClass1();
ConcreateClass2 cls2 = new ConcreateClass2();
cls1.templeteMethod();
cls2.templeteMethod();
}
}
扩展-钩子函数
abstract class AbstractHookClass{
//基本方法
protected abstract void doSomething();
//基本方法
protected abstract void doAnything();
//模板方法,注意:为了防止恶意的操作,一遍模板方法都加上final关键字,不允许被覆写。
public void templeteMethod(){
if(isDoSomething()){
this.doSomething();
}
this.doAnything();
}
/**
* 钩子函数(Hook Method)
*
* isDoSomething()的返回值影响了模板方法templeteMethod的执行结果
* @return
*/
protected boolean isDoSomething(){
return true;
}
}