模板模式 ( Template Pattern )
模板模式(Template Pattern)中,一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行
模板模式属于行为型模式
摘要
-
意图:
定义一个操作中的算法的骨架,而将一些步骤延迟到子类中
模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤
-
主要解决:
一些方法通用,却在每一个子类都重新写了这一方法
-
何时使用:
有一些通用的方法
-
如何解决:
将这些通用算法抽象出来
-
关键代码:
在抽象类实现,其他步骤在子类实现
-
应用实例:
- 在造房子的时候,地基、走线、水管都一样,只有在建筑的后期才有加壁橱加栅栏等差异
- 西游记里面菩萨定好的 81 难,这就是一个顶层的逻辑骨架
- Spring 中对 Hibernate 的支持,将一些已经定好的方法封装起来,比如开启事务、获取 Session、关闭 Session 等,程序员不重复写那些已经规范好的代码,直接丢一个实体就可以保存
-
优点:
- 封装不变部分,扩展可变部分
- 提取公共代码,便于维护
- 行为由父类控制,子类实现
-
缺点:
每一个不同的实现都需要一个子类来实现,导致类的个数增加,使得系统更加庞大
-
使用场景:
- 有多个子类共有的方法,且逻辑相同
- 重要的、复杂的方法,可以考虑作为模板方法
-
注意事项:
为防止恶意操作,一般模板方法都加上 final 关键词
实现
- 创建一个定义操作的 Game 抽象类,其中,模板方法设置为 final,这样它就不会被重写
- 定义类 Cricket 和 Football 扩展 Game,重写了抽象类的方法
- 定义类 TemplatePatternDemo 使用 Game 来演示模板模式的用法
范例
1. 创建一个抽象类,它的模板方法被设置为 final
Game.java
1 | public abstract class Game { |
2. 创建扩展了上述类的实体类
Cricket.java
1 | public class Cricket extends Game { |
Football.java
1 | public class Football extends Game { |
3. 使用 Game 的模板方法 play() 来演示游戏的定义方式
TemplatePatternDemo.java
1 | public class TemplatePatternDemo { |
编译运行以上 Java 范例,输出结果如下
1 | javac -d . src/main/cn/twle/gof/TemplatePatternDemo.java |