当前位置: 首页 > news >正文

对模板方法模式的理解

对模板方法模式的理解

    • 一、场景
      • 1、题目【[来源](https://kamacoder.com/problempage.php?pid=1087)】
        • 1.1 题目描述
        • 1.2 输入描述
        • 1.3 输出描述
        • 1.4 输入示例
        • 1.5 输出示例
    • 二、不采用模板方法模式
      • 1、代码
      • 2、问题
    • 三、采用模板方法模式
      • 1、代码
    • 四、总结

一、场景

1、题目【来源】

1.1 题目描述

小明喜欢品尝不同类型的咖啡,她发现每种咖啡的制作过程有一些相同的步骤,他决定设计一个简单的咖啡制作系统,使用模板方法模式定义咖啡的制作过程。系统支持两种咖啡类型:美式咖啡(American Coffee)和拿铁(Latte)。

咖啡制作过程包括以下步骤:

  1. 研磨咖啡豆 Grinding coffee beans
  2. 冲泡咖啡 Brewing coffee
  3. 添加调料 Adding condiments

其中,美式咖啡和拿铁的调料添加方式略有不同, 拿铁在添加调料时需要添加牛奶Adding milk

1.2 输入描述

多行输入,每行包含一个数字,表示咖啡的选择(1 表示美式咖啡,2 表示拿铁)。

1.3 输出描述

根据每行输入,输出制作咖啡的过程,包括咖啡类型和各个制作步骤,末尾有一个空行。

1.4 输入示例
1
2
1.5 输出示例
Making American Coffee:
Grinding coffee beans
Brewing coffee
Adding condiments

Making Latte:
Grinding coffee beans
Brewing coffee
Adding milk
Adding condiments

二、不采用模板方法模式

1、代码

public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        while (scanner.hasNextInt()) {
            int n = scanner.nextInt();

            if (n == 1) {
                System.out.println("Making American Coffee:");
                System.out.println("Grinding coffee beans");
                System.out.println("Brewing coffee");
                System.out.println("Adding condiments");


            } else if (n == 2) {
                System.out.println("Making Latte:");
                System.out.println("Grinding coffee beans");
                System.out.println("Brewing coffee");
                System.out.println("Adding milk");
                System.out.println("Adding condiments");

            } else {
                throw new RuntimeException("must be 1 or 2");
            }

            System.out.println();

        }
    }
}

2、问题

  • 虽然是制作不同的咖啡,但制作过程中,有些步骤是一样的。因此,可以复用这些步骤,减少代码冗余。
  • 咖啡是抽象的,但美式咖啡、拿铁是具体的。很容易想到”抽象类-实现子类“这样的代码结构。在抽象类中规范步骤,子类去覆写细节。

三、采用模板方法模式

1、代码

  • 模板
public interface Coffee {
    CoffeeEnum gotCoffeeEnum();
    void create();
}

public abstract class AbstractCoffee implements Coffee {
    @Override
    public final void create() {
        System.out.printf("Making %s:%n", gotCoffeeEnum().getDesc());
        System.out.println("Grinding coffee beans");
        System.out.println("Brewing coffee");
        addCondiments();
    }

    protected void addCondiments() {
        System.out.println("Adding condiments");
    }
}
  • 实现类
@Getter
@AllArgsConstructor
public enum CoffeeEnum {
    AMERICAN_COFFEE("American Coffee", 1),
    LATTE("Latte", 2)
    ;

    private final String desc;
    private final int code;

    public static CoffeeEnum getCoffeeEnum(int code) {
        for (CoffeeEnum coffeeEnum : CoffeeEnum.values()) {
            if (coffeeEnum.getCode() == code) {
                return coffeeEnum;
            }
        }
        return null;
    }
}

public class AmericanCoffee extends AbstractCoffee {
    @Override
    public CoffeeEnum gotCoffeeEnum() {
        return CoffeeEnum.AMERICAN_COFFEE;
    }
}


public class LatteCoffee extends AbstractCoffee {
    @Override
    public CoffeeEnum gotCoffeeEnum() {
        return CoffeeEnum.LATTE;
    }

    @Override
    protected void addCondiments() {
        System.out.println("Adding milk");
        System.out.println("Adding condiments");
    }
}
  • 外观模式,对各种Coffee的实现类进行封装:
public class CoffeeFacade {
    private List<Coffee> coffees;

    public CoffeeFacade() {
        this.coffees = new ArrayList<>();
        this.coffees.add(new AmericanCoffee());
        this.coffees.add(new LatteCoffee());
    }

    public void makeCoffee(int code) {
        coffees.stream()
                .filter(coffee -> coffee.gotCoffeeEnum().getCode() == code)
                .findFirst()
                .ifPresent(Coffee::create);
    }
}
  • 客户端:
public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        CoffeeFacade coffeeFacade = new CoffeeFacade();

        while (scanner.hasNextInt()) {
            int n = scanner.nextInt();
            coffeeFacade.makeCoffee(n);
            System.out.println();
        }

    }
}

四、总结

  • 在企业开发中,模板方法模式还是很常见的。通过分析需求,发现类与类的关系呈现下图所示,就可以在抽象类中定义通用步骤(模板),子类去覆写具体细节即可。

    image

  • 当然了,模板方法模式也存在一个明显的弊端:一旦某个实现类不适用当前模板了,不得不去修改抽象类的通用步骤时,便会影响现有实现类(复用的代价)。

  • 比较好的做法是,定义通用步骤时,步骤要足够抽象,而不能太具体太细了,这样可以给子类足够的灵活性。

http://www.dtcms.com/a/111207.html

相关文章:

  • WPF设计学习记录滴滴滴6
  • 池化技术的深度解析与实践指南【大模型总结】
  • 【51单片机】2-6【I/O口】电动车简易防盗报警器实现
  • Python循环控制语句
  • 幻觉抵抗优化大模型:teapotllm
  • Linux 线程1-线程的概念、线程与进程区别、线程的创建、线程的调度机制、线程函数传参
  • SpringBoot+Spring+MyBatis相关知识点
  • MQL5教程 05 指标开发实战:双色线、双线变色MACD、跨时间周期均线
  • TSMaster在新能源汽车研发测试中的硬核应用指南
  • 【rockchip】使用RKMPP+RGA解码H264并转换数据格式输出
  • 一文理解什么是中值模糊
  • C++多线程函数介绍
  • 【Kafka基础】ZooKeeper在Kafka中的核心作用:分布式系统中枢神经系统
  • 【LeetCode Solutions】LeetCode 141 ~ 145 题解
  • RocketMQ 中的 ProducerManager 组件剖析
  • 【Java Stream详解】
  • 提高:图论:强连通分量 图的遍历
  • Nginx功能及应用全解:从负载均衡到反向代理的全面剖析
  • OpenAI:人工智能领域的探索者与变革者
  • 黑马点评redis改 part 1
  • T-SQL语言的链表查找
  • eventEmitter实现
  • 网络建设与运维神州数码DCN MAC地址表操作
  • TypedDict和dataclass的优缺点对比
  • 前馈控制与反馈控制融合算法详解及python案例分析
  • JavaWeb学习--MyBatis-Plus整合SpringBoot的ServiceImpl方法(增加,修改与删除部分)
  • 深入解析:使用Python爬取Bilibili视频
  • 如何用DeepSeek进行SWOT分析?以CSDN的“C知道”为例
  • k8s的StorageClass存储类和pv、pvc、provisioner、物理存储的链路
  • 做一个Andriod系统应用的方法