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

当模板方法模式遇上工厂模式:一道优雅的烹饪架构设计

当模板方法模式遇上工厂模式:一道优雅的烹饪架构设计

  • 模式交响曲的实现
    • 模板方法模式
      • 搭建烹饪骨架(抽象类)
      • 具体菜品(子类)
    • 工厂模式
  • 模式协作的优势呈现
    • 扩展性演示
    • 运行时流程控制
  • 完整代码

在这里插入图片描述

如果在学习 设计模式的过程中,对模板方法模式知识点稍感生疏,记忆有些模糊不清了,不妨参考一下我的上一篇文章:【从厨房到代码——揭秘模板方法模式的烹饪哲学】相信能为你答疑解惑,助你重拾清晰记忆。
庖丁解牛看模式
在软件开发领域,我们经常面临需要标准化流程又保留扩展灵活性的场景。本文通过一个生动的厨房烹饪案例,展示如何模板方法模式工厂模式优雅结合,实现流程标准化与对象创建解耦的完美平衡。

模式交响曲的实现

模板方法模式

搭建烹饪骨架(抽象类)

我们首先定义抽象基类AbstractCooking,它采用三层结构设计
伪代码:

public abstract class AbstractCooking {// 模板方法(final确保流程不可篡改)public final void execute() { /* 标准流程 */ }// 固定步骤(私有方法)private void prepareIngredients() { /* ... */ }// 可选步骤(钩子方法)protected boolean needThickening() { return true; }// 抽象方法(必须实现)protected abstract void aromaBlasting();
}

具体代码:

public abstract class AbstractCooking {protected CookEnum cookEnum;// 模板方法(final锁定流程)public final void execute() {prepareIngredients();heatOil();aromaBlasting();  // 抽象方法:爆香方式mainCooking();    // 抽象方法:主料烹制if(needThickening()) thicken(); // 钩子控制收汁}private void prepareIngredients() {System.out.println("准备食材以及配料");}private void heatOil() {System.out.println("热锅凉油至180℃");}private void thicken() {System.out.println("收汁");}// 钩子方法:默认需要收汁protected boolean needThickening() {return true;}protected abstract void aromaBlasting();protected abstract void mainCooking();public CookEnum getCookEnum() {return cookEnum;}
}

这种设计实现了:

  • 流程固化:通过final方法锁定烹饪顺序
  • 灵活扩展:抽象方法强制子类实现关键步骤
  • 可选扩展:钩子方法提供流程微调能力

具体菜品(子类)

宫保鸡丁

/*** 宫保鸡丁*/
@Service
public class KungPaoChicken extends AbstractCooking {public KungPaoChicken() {this.cookEnum = CookEnum.KUNG_PAO_CHICKEN;}@Overrideprotected void aromaBlasting() {System.out.println("葱姜蒜爆香");}@Overrideprotected void mainCooking() {System.out.println("胡萝卜、鸡肉、花生切丁爆炒");}@Overrideprotected boolean needThickening() {    // 不需要收汁return false;}
}

麻婆豆腐

/*** 麻婆豆腐*/
@Service
public class MapoTofu  extends AbstractCooking {public MapoTofu() {this.cookEnum = CookEnum.MAPO_TO_FU;}@Overrideprotected void aromaBlasting() {System.out.println("煸炒郫县豆瓣酱+花椒粒");}@Overrideprotected void mainCooking() {System.out.println("嫩豆腐切块入红汤慢炖");}
}

工厂模式

CookFactory通过Spring容器自动装载所有菜品实现

@Service
public class CookFactory implements InitializingBean {@Autowiredprivate List<AbstractCooking> cookings;private Map<CookEnum, AbstractCooking> cookingMap = new HashMap<>();public AbstractCooking getCookingByCode(String code) {CookEnum cookEnum = CookEnum.getByCode(code);return cookingMap.get(cookEnum);}@Overridepublic void afterPropertiesSet() throws Exception {for (AbstractCooking cooking: cookings) {cookingMap.put(cooking.getCookEnum(), cooking);}}
}
/*** 菜品枚举类*/
public enum CookEnum {KUNG_PAO_CHICKEN("kungPaoChicken", "宫保鸡丁"),MAPO_TO_FU("mapoTofu", "麻婆豆腐");private final String code;private final String name;CookEnum(String code, String name) {this.code = code;this.name = name;}public String getCode() {return code;}public String getName() {return name;}public static CookEnum getByCode(String code) {for (CookEnum cookEnum : CookEnum.values()) {if (cookEnum.code.equals(code)) {return cookEnum;}}return null;}
}

这种实现方式具备:

  • 自动发现:利用Spring的依赖注入特性
  • 集中管理:统一维护菜品实例映射关系
  • 解耦调用:客户端无需关心具体实现类

模式协作的优势呈现

扩展性演示

@Service
public class BraisedPork extends AbstractCooking {// 实现抽象方法@Overrideprotected void aromaBlasting() {System.out.println("葱姜蒜大料爆香");}@Overrideprotected void mainCooking() {System.out.println("三层肉文火慢炖40分钟");}// 扩展枚举CookEnum.BRAISED_PORK
}

无需修改已有代码即可完成功能扩展,充分体现开闭原则

运行时流程控制

控制器通过简单调用实现完整烹饪流程

@GetMapping("/order")
public void cookDish(String dishCode) {cookFactory.getCooking(dishCode).execute();
}

完整代码

抽象类:

public abstract class AbstractCooking {protected CookEnum cookEnum;// 模板方法(final锁定流程)public final void execute() {prepareIngredients();heatOil();aromaBlasting();  // 抽象方法:爆香方式mainCooking();    // 抽象方法:主料烹制if(needThickening()) thicken(); // 钩子控制收汁}private void prepareIngredients() {System.out.println("准备食材以及配料");}private void heatOil() {System.out.println("热锅凉油至180℃");}private void thicken() {System.out.println("收汁");}// 钩子方法:默认需要收汁protected boolean needThickening() {return true;}protected abstract void aromaBlasting();protected abstract void mainCooking();public CookEnum getCookEnum() {return cookEnum;}
}

子类 - 宫保鸡丁:

/*** 宫保鸡丁*/
@Service
public class KungPaoChicken extends AbstractCooking {public KungPaoChicken() {this.cookEnum = CookEnum.KUNG_PAO_CHICKEN;}@Overrideprotected void aromaBlasting() {System.out.println("葱姜蒜爆香");}@Overrideprotected void mainCooking() {System.out.println("胡萝卜、鸡肉、花生切丁爆炒");}@Overrideprotected boolean needThickening() {    // 不需要收汁return false;}
}

子类 - 麻婆豆腐:

/*** 麻婆豆腐*/
@Service
public class MapoTofu  extends AbstractCooking {public MapoTofu() {this.cookEnum = CookEnum.MAPO_TO_FU;}@Overrideprotected void aromaBlasting() {System.out.println("煸炒郫县豆瓣酱+花椒粒");}@Overrideprotected void mainCooking() {System.out.println("嫩豆腐切块入红汤慢炖");}
}

工厂类:

@Service
public class CookFactory implements InitializingBean {@Autowiredprivate List<AbstractCooking> cookings;private Map<CookEnum, AbstractCooking> cookingMap = new HashMap<>();public AbstractCooking getCookingByCode(String code) {CookEnum cookEnum = CookEnum.getByCode(code);return cookingMap.get(cookEnum);}@Overridepublic void afterPropertiesSet() throws Exception {for (AbstractCooking cooking: cookings) {cookingMap.put(cooking.getCookEnum(), cooking);}}
}

枚举:

/*** 菜品枚举类*/
public enum CookEnum {KUNG_PAO_CHICKEN("kungPaoChicken", "宫保鸡丁"),MAPO_TO_FU("mapoTofu", "麻婆豆腐");private final String code;private final String name;CookEnum(String code, String name) {this.code = code;this.name = name;}public String getCode() {return code;}public String getName() {return name;}public static CookEnum getByCode(String code) {for (CookEnum cookEnum : CookEnum.values()) {if (cookEnum.code.equals(code)) {return cookEnum;}}return null;}
}

接口:

@RestController
@RequestMapping("/test")
public class Testcontroller {@AutowiredCookFactory cookFactory;@GetMapping("/orderDishes")public void orderDishes(String dishCode) {cookFactory.getCookingByCode(dishCode).execute();}
}

相关文章:

当模板方法模式遇上工厂模式:一道优雅的烹饪架构设计

当模板方法模式遇上工厂模式&#xff1a;一道优雅的烹饪架构设计 模式交响曲的实现模板方法模式搭建烹饪骨架&#xff08;抽象类&#xff09;具体菜品&#xff08;子类&#xff09; 工厂模式 模式协作的优势呈现扩展性演示运行时流程控制 完整代码 如果在学习 设计模式的过程中…...

c++位运算总结

在C中&#xff0c;位运算是对二进制位进行操作的运算&#xff0c;主要有以下几种&#xff1a; 1. 按位与&#xff08; & &#xff09;&#xff1a;两个操作数对应位都为1时&#xff0c;结果位才为1&#xff0c;否则为0。例如 3 & 5 &#xff0c; 3 二进制是 0000 0011…...

企业级知识库建设:自建与开源产品集成的全景解析 —— 产品经理、CTO 与 CDO 的深度对话

文章目录 一、引言二、主流产品与方案对比表三、自建方案 vs. 开源产品集成&#xff1a;技术路径对比3.1 自建方案3.2 开源产品集成方案 四、结论与个人观点 一、引言 在当今数据驱动的商业环境中&#xff0c;构建高质量的知识库已成为企业数字化转型的关键一环。本博客分别从…...

Python小练习系列 Vol.6:单词搜索(网格回溯)

&#x1f9e0; Python小练习系列 Vol.6&#xff1a;单词搜索&#xff08;网格回溯&#xff09; &#x1f50d; 本期我们来挑战一道 LeetCode 上经典的网格型回溯题 —— 单词搜索&#xff0c;考察对 DFS 状态恢复的掌握&#xff01; &#x1f9e9; 一、题目描述 给定一个 m x…...

shell脚本--MySQL简单调用

实现功能 增 数据库的创建&#xff0c;数据表的创建已经实现 创建用户 删 删除数据库&#xff0c; 删除库下的某个表&#xff0c; 删除某个用户 改 暂无 查 查看所有的数据库&#xff0c; 查看某个库下的所有数据表&#xff0c; 查看某个表的结构&#xff0c; 查…...

vue3项目配置别名

vue3项目配置别名 src别名的配置TypeScript 编译配置如果出现/别名引入报找不到的问题 src别名的配置 在开发项目的时候文件与文件关系可能很复杂&#xff0c;因此我们需要给src文件夹配置一个别名&#xff01;&#xff01;&#xff01; // vite.config.ts import {defineCon…...

Rust 面向对象

Rust 面向对象 引言 Rust 是一种系统编程语言,以其高性能、内存安全和并发支持而受到关注。Rust 的面向对象特性是其强大功能之一,它允许开发者以面向对象的方式构建复杂的应用程序。本文将深入探讨 Rust 的面向对象编程(OOP)特性,包括类的定义、继承、封装和多态等概念…...

[ C语言 ] | 从0到1?

目录 认识计算机语言 C语言 工欲善其事必先利其器 第一个C语言代码 这一些列 [ C语言 ] &#xff0c;就来分享一下 C语言 相关的知识点~ 认识计算机语言 我们说到计算机语言&#xff0c;语言&#xff0c;就是用来沟通的工具&#xff0c;计算机语言呢&#xff1f;就是我们…...

[Mac]利用Hexo+Github Pages搭建个人博客

由于我这台Mac基本没啥环境&#xff0c;因此需要从零开始配置&#xff0c;供各位参考。 注意⚠️&#xff1a;MacBook (M4)使用/bin/zsh作为默认Shell&#xff0c;其对应的配置文件为~/.zshrc 参考文档&#xff1a; HEXO系列教程 | 使用GitHub部署静态博客HEXO | 小白向教程 文…...

pycharm与python版本

python 3.6-3.9 pycharm 2021版本搭配最好 python 3.8 pycharm 2019版本搭配最好 pycharm各版本下载...

Qt在IMX6ULL嵌入式系统中图片加载问题排查与解决

Qt在IMX6ULL嵌入式系统中图片加载问题排查与解决&#xff08;保姆级教学&#xff01;&#xff09; 在使用Qt开发IMX6ULL嵌入式系统的过程中&#xff0c;我遇到了图片加载的常见问题。本文将分享问题排查的详细过程和解决方案&#xff0c;希望能帮助遇到类似困难的开发者。 问题…...

界面控件Telerik和Kendo UI 2025 Q1亮点——AI集成与数据可视化

Telerik DevCraft包含一个完整的产品栈来构建您下一个Web、移动和桌面应用程序。它使用HTML和每个.NET平台的UI库&#xff0c;加快开发速度。Telerik DevCraft提供完整的工具箱&#xff0c;用于构建现代和面向未来的业务应用程序&#xff0c;目前提供UI for ASP.NET MVC、Kendo…...

pycharm终端操作远程服务器

pycharm项目已经连接了远程服务器&#xff0c;但是打开终端&#xff0c;却依旧显示的是本地的那个环境&#xff0c;也就是说没有操作远程的那个环境。只能再使用Xshell去操作远程环境&#xff0c;很麻烦&#xff0c;找了下教程。 来源&#xff1a;https://blog.csdn.net/maolim…...

接口测试中数据库验证,怎么解决?

在接口测试中&#xff0c;通常需要在接口调用前后查询数据库&#xff0c;以验证接口操作是否正确影响了数据库状态。​这可以通过数据库断言来实现&#xff0c;PyMySQL库常用于连接和操作MySQL数据库。​通过该库&#xff0c;可以在测试中执行SQL语句&#xff0c;查询或修改数据…...

Playwright从入门到实战:比Selenium更快的数据爬取案例实战

摘要 Playwright 是微软开源的下一代浏览器自动化工具&#xff0c;凭借其高性能、跨浏览器支持和现代化设计&#xff0c;迅速成为 Web 自动化领域的热门选择。本文将从 安装配置 开始&#xff0c;通过 实战演练 展示其核心功能&#xff0c;并与 Selenium 深度对比&#xff0c;…...

defconfig配置宏的规则

defconfig配置宏的规则 CONFIG_INETnCONFIG_INETy defconfig里这样配置&#xff0c;CONFIG_INET宏有效吗 在 defconfig 文件中&#xff0c;如果出现了 相同的配置项被定义多次&#xff0c;最终生效的是最后一次出现的值。 &#x1f539; 你的配置 bash复制编辑CONFIG_INE…...

day1_Flink基础

文章目录 Flink基础今日课程内容目标为什么要学Flink技术更新迭代市场需求 流式计算批量计算概念特点 批量计算的优势和弊端流式计算生活中流场景流式计算的概念 Flink简介Flink历史Flink介绍 Flink架构体系已学过的框架技术Flink架构 Flink集群搭建Flink的集群模式Standalone模…...

ctf-web: 不统一的解析 + sql注入要求输入与输出相等 -- tpctf supersqli

# 从 django.shortcuts 模块导入 render 函数&#xff0c;用于渲染模板 from django.shortcuts import render # 从 django.db 模块导入 connection 对象&#xff0c;用于数据库连接 from django.db import connection# 此模块用于创建视图函数 # 从 django.http 模块导入 Http…...

基于Java与Go的下一代DDoS防御体系构建实战

引言:混合云时代的攻防对抗新格局 2024年某金融平台遭遇峰值2.3Tbps的IPv6混合攻击,传统WAF方案在新型AI驱动攻击面前全面失效。本文将以Java与Go为技术栈,揭示如何构建具备智能决策能力的防御系统。 一、攻击防御技术矩阵重构 1.1 混合攻击特征识别 攻击类型Java检测方案…...

使用FastExcel时的单个和批量插入的问题

在我们用excel表进行插入导出的时候&#xff0c;通常使用easyexcel或者FastExcel&#xff0c;而fastexcel是easy的升级版本&#xff0c;今天我们就对使用FastExcel时往数据库插入数据的业务场景做出一个详细的剖析 场景1 现在我们数据库有一张组织表&#xff0c;组织表的字段…...

交换技术综合实验

一、实验拓扑 二、实验要求 内网IP地址使用172.16.0.0/16分配。 SW1和SW2之间互为备份。 VRRP/STP/VLAN/Eth-trunk均使用。 所有PC通过DHCP获取IP地址。 ISP只能配置IP地址。 所有电脑可以正常访问ISP路由器。 三、实验步骤 基于172.16.0.0/16进行划分 172.16.2.0/24&…...

软件工程之软件开发模型(瀑布、迭代、敏捷、DevOps)

1. 瀑布模型&#xff08;Waterfall Model&#xff09; 定义与流程 瀑布模型是线性顺序的开发流程&#xff0c;包含需求分析、设计、编码、测试、维护等阶段&#xff0c;每个阶段完成后才能进入下一阶段&#xff0c;类似“瀑布流水”逐级推进。 核心特点 严格阶段划分&#…...

Display Serializer、Camera Deserializer(Camera Des)和SerDes‌ 加解串应用

‌1. 概述&#xff1a;三者的核心定位‌ ‌(1) SerDes&#xff08;Serializer/Deserializer&#xff09;‌ ‌定义‌&#xff1a;通用高速数据传输技术&#xff0c;实现‌并行↔串行‌双向转换。‌角色‌&#xff1a;数据链路的“翻译官”&#xff0c;解决并行传输的带宽与距…...

Redis 常用数据结构及其对应的业务场景(总结)

1. String&#xff08;字符串&#xff09; 特点&#xff1a;最简单的键值对结构&#xff0c;可存储文本、数字或二进制数据&#xff08;最大 512MB&#xff09;。 适用场景&#xff1a; 缓存&#xff1a;存储用户信息、页面片段、商品详情等&#xff08;如 SET user:1 "{…...

记录Jmeter 利用BeanShell 脚本解析JSON字符串

下载org.json包(文档说明) #下载地址 https://www.json.org/ # github 地址 https://github.com/stleary/JSON-java # api 文档说明 https://resources.arcgis.com/en/help/arcobjects-java/api/arcobjects/com/esri/arcgis/server/json/JSONObject.htmlBeanShell脚本 import…...

深入解析音频:格式、同步及封装容器

物理音频和数字音频 物理音频 定义&#xff1a;物理音频就是声音在自然界中的物理表现形式&#xff0c;本质上是一种机械波&#xff0c;通过空气或其他介质传播。例如&#xff0c;当我们说话、乐器演奏或物体碰撞时&#xff0c;都会产生振动&#xff0c;这些振动会引起周围介…...

RPCGC阅读

24年的MM 创新 现有点云压缩工作主要集中在保真度优化上。 而在实际应用中&#xff0c;压缩的目的是促进机器分析。例如&#xff0c;在自动驾驶中&#xff0c;有损压缩会显着丢失户外场景的详细信息。在三维重建中&#xff0c;压缩过程也会导致场景数据中语义信息(Contour)的…...

医疗CMS高效管理:简化更新维护流程

内容概要 医疗行业内容管理系统&#xff08;CMS&#xff09;的核心价值在于应对医疗信息管理的多维复杂性。面对诊疗指南的动态更新、科研数据的快速迭代以及多机构协作需求&#xff0c;传统管理模式往往面临效率瓶颈与合规风险。现代化医疗CMS通过构建结构化权限管理矩阵&…...

《Spring Cloud Eureka 高可用集群实战:从零构建高可靠性的微服务注册中心》

从零构建高可用 Eureka 集群 | Spring Cloud 微服务架构深度实践指南 本文核心内容基于《Spring Cloud 微服务架构开发》第1版整理&#xff0c;结合生产级实践经验优化 实验环境&#xff1a;IntelliJ IDEA 2024 | JDK 1.8| Spring Boot 2.1.7.RELEASE | Spring Cloud Greenwich…...

PyQt6实例_批量下载pdf工具_主线程启用线程池

目录 前置&#xff1a; 代码&#xff1a; 视频&#xff1a; 前置&#xff1a; 1 本系列将以 “PyQt6实例_批量下载pdf工具”开头&#xff0c;放在 【PyQt6实例】 专栏 2 本系列涉及到的PyQt6知识点&#xff1a; 线程池&#xff1a;QThreadPool,QRunnable&#xff1b; 信号与…...