Java设计模式之模板方法模式

概念

模板方法模式是一种行为型设计模式,它定义了一个算法的骨架,允许子类在不改变算法结构的情况下重定义某些步骤。父类通过模板方法控制流程,子类通过实现抽象方法填充细节。

作用

**1.代码复用:**公共逻辑抽取到父类,避免重复代码。

**2.扩展性:**子类只需关注差异部分,符合开闭原则。

**3.流程控制:**父类固定核心流程,防止子类破坏执行顺序。

场景

1.多个类具有相似流程但部分步骤不同(如数据解析、文件处理)。

2.需要固定核心算法流程,允许子类扩展特定步骤。

3.框架设计(如Spring的JdbcTemplate)。

举例(饮料制作)

复制代码
// 饮料类
abstract class Beverage {
    // 准备食谱:模板方法 (final防止子类覆盖)
    public final void prepareRecipe() {
        // 烧开水
        boilWater();
        // 泡
        brew();
        // 倒入杯子
        pourInCup();
        // 是否加入调味剂
        if (customerWantsCondiments()) {
            addCondiments();
        }
    }

    // 固定步骤:烧开水
    private void boilWater() {
        System.out.println("烧开水");
    }
    // 固定步骤:倒入杯子
    private void pourInCup() {
        System.out.println("倒入杯子");
    }

    // 抽象方法(必须实现)
    abstract void brew();
    abstract void addCondiments();

    // 钩子方法(可选覆盖)
    boolean customerWantsCondiments() {
        return true;
    }
}

class Coffee extends Beverage {
    @Override
    void brew() {
        System.out.println("冲泡咖啡粉");
    }

    @Override
    void addCondiments() {
        System.out.println("加糖和牛奶");
    }

    @Override
    boolean customerWantsCondiments() {
        return false; // 示例:咖啡不要调料
    }
}

class Tea extends Beverage {
    @Override
    void brew() {
        System.out.println("浸泡茶叶");
    }

    @Override
    void addCondiments() {
        System.out.println("加柠檬");
    }
}

优点和缺点

|----------------|------------------|
| 优点 | 缺点 |
| 代码复用率高 | 增加类数量(每个差异需一个子类) |
| 扩展性强(新增子类即可) | 父类修改可能影响所有子类 |
| 反向控制结构(父类调用子类) | 过度继承可能导致系统复杂 |

不使用模板方法的实现

复制代码
class CoffeeWithoutPattern {
    public void makeCoffee() {
        System.out.println("烧开水");
        System.out.println("冲泡咖啡粉");
        System.out.println("倒入杯子");
        System.out.println("加糖和牛奶"); // 重复代码
    }
}

class TeaWithoutPattern {
    public void makeTea() {
        System.out.println("烧开水");       // 重复代码
        System.out.println("浸泡茶叶");
        System.out.println("倒入杯子");    // 重复代码
        System.out.println("加柠檬");
    }
}

存在的问题

**1.重复代码:**烧开水、倒杯子的代码重复。

**2.维护困难:**修改流程需改动所有类。

**3.无法强制统一流程:**子类可能遗漏步骤或改变顺序。

|------|-----|-----|------|------|
| 方案 | 代码量 | 扩展性 | 维护成本 | 流程控制 |
| 模板方法 | 少 | 高 | 低 | 严格统一 |
| 传统实现 | 多 | 低 | 高 | 自由松散 |

设计启示:当遇到多个相似流程的类时,优先使用模板方法模式消除重复代码,尤其适合需要严格流程控制的场景。

相关推荐
Lyyaoo.3 分钟前
【JAVA基础面经】线程的状态
java·开发语言
Hello小赵3 分钟前
C语言如何自定义链接库——编译与调用
android·java·c语言
希望永不加班7 分钟前
SpringBoot 配置绑定:@ConfigurationProperties
java·spring boot·后端·spring
悟空码字8 分钟前
MySQL性能优化的天花板:10条你必须掌握的顶级SQL分析技巧
java·后端·mysql
indexsunny12 分钟前
互联网大厂Java面试实战:Spring Boot、MyBatis与Kafka在电商场景中的应用
java·spring boot·面试·kafka·mybatis·电商·技术栈
殷紫川16 分钟前
CompletableFuture 异步编程全解:核心能力、编排方案、异常处理与超时控制
java
ss27321 分钟前
致Java初学者的一封信
java·开发语言
white-persist21 分钟前
【vulhub spring CVE-2018-1270】CVE-2018-1270 Spring Messaging 远程命令执行漏洞 完整复现详细分析解释
java·服务器·网络·数据库·后端·python·spring
潇洒畅想29 分钟前
1.1 从∑到∫:用循环理解求和与累积
java·数据结构·python·算法
维齐洛波奇特利(male)1 小时前
@Pointcut(“execution(* com.hdzx..*(..))“)切入点与aop 导致无限循环
java·开发语言