17370845950

如何在Java中理解模板方法模式

模板方法模式的核心在于定义一个算法的骨架,而将一些步骤延迟到子类中实现。它让子类可以在不改变算法结构的前提下,重新定义算法中的某些步骤。在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,而不用关心整体顺序。

关键点与使用场景

模板方法模式的关键优势是代码复用和流程控制。父类控制算法结构,防止子类破坏流程,同时允许扩展。

  • 当多个类有相似的算法步骤,只是细节不同时,适合使用模板方法
  • 模板方法应声明为 final,防止子类修改整体流程
  • 钩子方法提供灵活性,让子类能影响父类的行为
  • 常用于框架设计,比如Spring中的JdbcTemplate、RestTemplate等底层都用了类似思想

基本上就这些。理解模板方法的重点是分清“变”与“不变”——把不变的逻辑放在父类,把可变的部分交给子类去实现。这样既保证结构统一,又具备扩展性。