模板方法模式

模板方法模式

文章目录

什么是模板方法模式

模板方法模式,定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤

重复 = 易错 + 难改

继承

我们既然用了继承,并且肯定这个继承有意义,就应该要成为子类的模板,所有重复的代码都应该要上升到父类去,而不是让每个子类都去重复。
模板方法登场,当我们要完成在某一细节层次一致的一个过程或一系列步骤,但其个别步骤在更详细的层次上的实现可能不同时,我们通常考虑用模板方法模式来处理
模板方法模式是通过把不变行为搬移到超类,去除子类中的重复代码来体现它的优势
模板方法模式就是提供了一个很好的代码复用平台
当不变的和可变的行为在方法的子类实现中混合在一起的时候,不变的行为就会在子类中重复出现。我们通过模板方法模式把这些行为搬移到单一的地方,这样就帮助子类摆脱重复的不变行为的纠缠

示例

模板方法模式是一种行为设计模式,它在一个抽象类中定义了一个算法的框架,允许子类在不改变结构的情况下重写算法的特定步骤。以下是一个使用 Java 实现模板方法模式的示例,以制作咖啡(Coffee)为例:

1.定义一个抽象的咖啡基类 Coffee,其中包含一个模板方法 makeCoffee(),它定义了制作咖啡的基本流程。在这个模板方法中,调用了若干个抽象方法(钩子方法),这些方法将在子类中实现具体的步骤:

java 复制代码
public abstract class Coffee {
    public final void makeCoffee() {
        boilWater();
        brewCoffeeGrinds();
        pourInCup();
        addCondiments();
    }

    // 钩子方法,子类可以选择是否覆盖
    protected void boilWater() {
        System.out.println("Boiling water...");
    }

    // 钩子方法,子类必须覆盖
    protected abstract void brewCoffeeGrinds();

    // 钩子方法,子类可以选择是否覆盖
    protected void pourInCup() {
        System.out.println("Pouring coffee into a cup...");
    }

    // 钩子方法,子类可以选择是否覆盖
    protected void addCondiments() {
        System.out.println("Adding sugar and milk...");
    }
}

2.创建具体的咖啡子类,如 Espresso 和 Latte,它们继承自 Coffee 类并实现或覆盖父类中的抽象方法:

java 复制代码
public class Espresso extends Coffee {
    @Override
    protected void brewCoffeeGrinds() {
        System.out.println("Brewing espresso...");
    }

    // 可选地,覆盖父类的默认实现
    @Override
    protected void addCondiments() {
        System.out.println("Adding a shot of vanilla syrup...");
    }
}
java 复制代码
public class Latte extends Coffee {
    @Override
    protected void brewCoffeeGrinds() {
        System.out.println("Brewing latte...");
    }

    // 可选地,覆盖父类的默认实现
    @Override
    protected void addCondiments() {
        System.out.println("Frothing milk and pouring it over the coffee...");
    }
}

3.在客户端代码中,创建具体的咖啡子类对象并调用模板方法 makeCoffee():

java 复制代码
public class CoffeeShop {
    public static void main(String[] args) {
        Coffee espresso = new Espresso();
        espresso.makeCoffee();

        System.out.println("\n---\n");

        Coffee latte = new Latte();
        latte.makeCoffee();
    }
}

运行 CoffeeShop 类的 main 方法,输出结果应为:

复制代码
Boiling water...
Brewing espresso...
Pouring coffee into a cup...
Adding a shot of vanilla syrup...

---

Boiling water...
Brewing latte...

总结

  1. 定义抽象类,并定义好具体的流程
  2. 具体的流程可以选择性的是否重写,从而达到具体某个步骤不通逻辑处理的实现
相关推荐
xieliyu.4 分钟前
Java手搓数据结构:从零模拟实现无头双向非循环链表
java·数据结构·链表
csbysj20204 分钟前
Java 条件语句
开发语言
薪火铺子23 分钟前
SpringMVC请求处理流程源码解析(第3篇):视图渲染与异常处理
java·后端·spring
Ulyanov41 分钟前
《现代 Python 桌面应用架构实战:PySide6 + QML 从入门到工程化》 开发环境搭建与工具链极简主义 —— 拒绝臃肿,构建工业级基座
开发语言·python·qt·ui·架构·系统仿真
逻辑驱动的ken1 小时前
Java高频面试场景题19
java·开发语言·面试·职场和发展·求职招聘
初心未改HD1 小时前
Go语言net/http与Web开发:构建高性能HTTP服务
开发语言·golang
leoufung1 小时前
LeetCode 42:接雨水 —— 从“矩形法”到双指针的完整思考过程
java·算法·leetcode
小碗羊肉1 小时前
【MySQL | 第十一篇】InnoDB引擎
java·数据库·mysql
Dylan的码园1 小时前
Maven基础架构与整体认识
java·junit·maven
叼烟扛炮1 小时前
C++第一讲:C++ 入门基础
开发语言·c++·函数重载·引用·内联函数·nullptr