
模板方法模式的核心在于定义一个算法的骨架,而将一些步骤延迟到子类中实现。它让子类可以在不改变算法结构的前提下,重新定义算法中的某些步骤。在Java中,这通常通过抽象类来实现。
模板方法的基本结构
模板方法一般定义在抽象类中,它是一个具体的方法,包含对多个基本方法的调用。这些基本方法可以是:
-
抽象方法:必须由子类实现
-
钩子方法(hook):有默认实现,子类可选择性地覆盖
-
具体方法:在整个流程中不变的部分,由父类完成
子类通过继承抽象类并实现抽象方法,来定制算法的特定部分。
一个简单的例子:制作饮品
public abstract class BeverageTemplate {
// 模板方法,定义流程
public final void prepare() {
boilWater();
brew();
pourInCup();
if (customerWantsCondiments()) {
addCondiments();
}
}
void boilWater() {
System.out.println("烧水");
}
abstract void brew(); // 冲泡,由子类实现
void pourInCup() {
System.out.println("倒入杯子");
}
abstract void addCondiments(); // 添加调料
// 钩子方法,默认返回true,子类可覆盖
boolean customerWantsCondiments() {
return true;
}
}
public class Coffee extends BeverageTemplate {
@Override
void brew() {
System.out.println("用咖啡机冲泡咖啡");
}
@Override
void addCondiments() {
System.out.println("加糖和牛奶");
}
}
public class Tea extends BeverageTemplate {
@Override
void brew() {
System.out.println("用热水冲泡茶叶");
}
@Override
void addCondiments() {
System.out.println("加柠檬");
}
// 覆盖钩子,控制流程
@Override
boolean customerWantsCondiments() {
return false; // 不想加调料
}
}
在这个例子中,prepare() 是模板方法,封装了制作饮品的整体流程。每个子类只需关注如何实现 brew 和 addCondiments,而不用关心整体顺序。
立即学习“Java免费学习笔记(深入)”;
关键点与使用场景
模板方法模式的关键优势是代码复用和流程控制。父类控制算法结构,防止子类破坏流程,同时允许扩展。
- 当多个类有相似的算法步骤,只是细节不同时,适合使用模板方法
- 模板方法应声明为 final,防止子类修改整体流程
- 钩子方法提供灵活性,让子类能影响父类的行为
- 常用于框架设计,比如Spring中的JdbcTemplate、RestTemplate等底层都用了类似思想
基本上就这些。理解模板方法的重点是分清“变”与“不变”——把不变的逻辑放在父类,把可变的部分交给子类去实现。这样既保证结构统一,又具备扩展性。
以上就是如何在Java中理解模板方法模式的详细内容,更多请关注php中文网其它相关文章!