模板方法模式通过抽象类定义算法骨架,子类实现具体步骤。示例中DataExporter定义export流程,子类CsvExporter和JsonExporter实现doExport,输出对应格式。钩子方法shouldValidate允许子类控制流程,如FastCsvExporter跳过验证。模板方法应声明为final,确保流程不被篡改,适用于算法结构固定、部分步骤变化的场景,提升代码复用与扩展性。

在Java中,模板方法模式是一种行为设计模式,它在抽象类中定义一个算法的骨架,将一些步骤延迟到子类中实现。模板方法使得子类可以在不改变算法结构的情况下重新定义算法的某些特定步骤。
定义抽象类与模板方法
模板方法通常是一个final方法,防止子类修改算法流程。而具体的步骤则由抽象方法或可重写的方法构成,交由子类实现。
以下是一个简单的示例:假设我们要处理不同类型的数据导出流程(如导出为CSV、JSON等),它们有共同的步骤:准备数据、验证数据、执行导出、记录日志。
public abstract class DataExporter { // 模板方法,定义算法骨架 public final void export() { prepareData(); validateData(); doExport(); // 实际导出由子类实现 logExport(); } protected void prepareData() { System.out.println("准备通用数据..."); } protected void validateData() { System.out.println("验证数据格式..."); } // 抽象方法,强制子类实现 protected abstract void doExport(); protected void logExport() { System.out.println("记录导出日志..."); } }子类实现具体步骤
子类继承抽象类并实现抽象方法,从而定制特定行为。
立即学习“Java免费学习笔记(深入)”;
使用时只需调用模板方法:
public class Main { public static void main(String[] args) { DataExporter csvExporter = new CsvExporter(); csvExporter.export(); DataExporter jsonExporter = new JsonExporter(); jsonExporter.export(); } } 输出结果:- 准备通用数据...
- 验证数据格式...
- 正在导出为CSV格式...
- 记录导出日志...
- 准备通用数据...
- 验证数据格式...
- 正在导出为JSON格式...
- 记录导出日志...
使用钩子方法控制流程
有时我们希望子类能影响算法流程,比如跳过某个步骤。这时可以使用“钩子方法”(Hook Method),它在抽象类中提供默认实现,子类可选择性地覆盖。
public abstract class DataExporter { public final void export() { prepareData(); if (shouldValidate()) { validateData(); } doExport(); logExport(); } protected void prepareData() { System.out.println("准备通用数据..."); } protected void validateData() { System.out.println("验证数据格式..."); } // 钩子方法,子类可决定是否执行验证 protected boolean shouldValidate() { return true; } protected abstract void doExport(); protected void logExport() { System.out.println("记录导出日志..."); } }子类可以选择跳过验证:
public class FastCsvExporter extends CsvExporter { @Override protected boolean shouldValidate() { return false; // 跳过验证 } }实践建议与注意事项
使用模板方法时需注意以下几点:
- 模板方法本身应声明为 final,避免子类篡改整体流程。
- 将变化的部分提取为抽象方法,不变的部分在抽象类中实现。
- 合理使用钩子方法增加灵活性,但不要过度复杂化流程判断。
- 适用于多个子类有相同算法结构,仅部分步骤不同的场景。
- 避免抽象类中依赖子类未实现的状态,防止空指针等问题。










