模板方法模式

模板方法模式

文章目录

什么是模板方法模式

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

重复 = 易错 + 难改

继承

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

示例

模板方法模式是一种行为设计模式,它在一个抽象类中定义了一个算法的框架,允许子类在不改变结构的情况下重写算法的特定步骤。以下是一个使用 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. 具体的流程可以选择性的是否重写,从而达到具体某个步骤不通逻辑处理的实现
相关推荐
蓝澈112112 分钟前
迪杰斯特拉算法之解决单源最短路径问题
java·数据结构
Kali_0719 分钟前
使用 Mathematical_Expression 从零开始实现数学题目的作答小游戏【可复制代码】
java·人工智能·免费
rzl0231 分钟前
java web5(黑马)
java·开发语言·前端
jingling55543 分钟前
面试版-前端开发核心知识
开发语言·前端·javascript·vue.js·面试·前端框架
guojl1 小时前
深度解读jdk8 HashMap设计与源码
java
guojl1 小时前
深度解读jdk8 ConcurrentHashMap设计与源码
java
m0_687399841 小时前
写一个Ununtu C++ 程序,调用ffmpeg API, 来判断一个数字电影的视频文件mxf 是不是Jpeg2000?
开发语言·c++·ffmpeg
爱上语文1 小时前
Redis基础(5):Redis的Java客户端
java·开发语言·数据库·redis·后端
A~taoker1 小时前
taoker的项目维护(ng服务器)
java·开发语言
萧曵 丶1 小时前
Rust 中的返回类型
开发语言·后端·rust