模板方法模式

模板方法模式

文章目录

什么是模板方法模式

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

重复 = 易错 + 难改

继承

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

示例

模板方法模式是一种行为设计模式,它在一个抽象类中定义了一个算法的框架,允许子类在不改变结构的情况下重写算法的特定步骤。以下是一个使用 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. 具体的流程可以选择性的是否重写,从而达到具体某个步骤不通逻辑处理的实现
相关推荐
z落落6 分钟前
C# 构造函数(无参/有参/重载/this)+析构函数(终结器)|GC 垃圾回收
java·开发语言·c#
武子康6 分钟前
Java-12 深入浅出 MyBatis 二级缓存详解:跨 SqlSession 共享与失效机制
java·后端
考虑考虑15 分钟前
JDK9中的Set.of()使用注意
java·后端·java ee
plainGeekDev18 分钟前
findViewById → ViewBinding
java·kotlin·gradle
kkeeper~22 分钟前
0基础C语言积跬步之自定义类型结构体
c语言·开发语言
yz_aiks25 分钟前
IDEA终端配置oh-my-zsh实战:安装、插件与日常使用技巧
java·ide·intellij-idea
Refrain_zc26 分钟前
Android 老设备存储空间展示:机身存储 + TF 卡容量获取完整实现
java
java1234_小锋31 分钟前
LangChain4j 开发Java Agent智能体- HelloWorld 实现
java·langchain4j
z落落32 分钟前
C# 字段与属性(get/set访问器、三种属性写法、只读属性)+属性拦截例子(get动态计算 + set数据校验)
开发语言·c#
RainCity34 分钟前
Java Swing 自定义组件库分享(十)
java·笔记·后端