在模板模式(Template Pattern)中,一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。这种类型的设计模式属于行为型模式。
介绍
意图: 定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。
主要解决: 一些方法通用,却在每一个子类都重新写了这一方法。
何时使用: 有一些通用的方法。
如何解决: 将这些通用算法抽象出来。
关键代码: 在抽象类实现,其他步骤在子类实现。
应用实例:
- 在造房子的时候,地基、走线、水管都一样,只有在建筑的后期才有加壁橱加栅栏等差异。
- 西游记里面菩萨定好的 81 难,这就是一个顶层的逻辑骨架。
- spring 中对 Hibernate 的支持,将一些已经定好的方法封装起来,比如开启事务、获取 Session、关闭 Session 等,程序员不重复写那些已经规范好的代码,直接丢一个实体就可以保存。
优点:
- 封装不变部分,扩展可变部分。
- 提取公共代码,便于维护。
- 行为由父类控制,子类实现。
缺点:每一个不同的实现都需要一个子类来实现,导致类的个数增加,使得系统更加庞大。
使用场景:
- 有多个子类共有的方法,且逻辑相同。
- 重要的、复杂的方法,可以考虑作为模板方法。
注意事项:为防止恶意操作,一般模板方法都加上 final 关键词。
代码
public abstract class Game {
/**
* login
*/
abstract void login();
/**
* start
*/
abstract void start();
/**
* end
*/
abstract void end();
/**
* 模板方法
*/
public final void play(){
login();
start();
end();
}
}
- 具体方法实现类
public class LOL extends Game {
/**
* login
*/
@Override
void login() {
System.out.println("登陆英雄联盟!");
}
/**
* start
*/
@Override
void start() {
System.out.println("欢迎进入召唤师峡谷!");
}
/**
* end
*/
@Override
void end() {
System.out.println("结束一场对局!");
}
}
- 具体方法实现类
public class CF extends Game {
/**
* login
*/
@Override
void login() {
System.out.println("登陆穿越火线!");
}
/**
* start
*/
@Override
void start() {
System.out.println("欢迎进入团队竞技频道!");
}
/**
* end
*/
@Override
void end() {
System.out.println("结束一场枪战!");
}
}
- 测试
public static void main(String[] args) {
Game game = new LOL();
game.play();
System.out.println();
Game game2 = new CF();
game2.play();
}
- 结果
登陆英雄联盟!
欢迎进入召唤师峡谷!
结束一场对局!
登陆穿越火线!
欢迎进入团队竞技频道!
结束一场枪战!
参考&转载
- 菜鸟教程:https://www.runoob.com/design-pattern/template-pattern.html