设计模式 Day 2:工厂方法模式(Factory Method Pattern)详解
继 Day 1 学习了单例模式之后,今天我们继续深入对象创建型设计模式——工厂方法模式(Factory Method)。工厂方法模式为对象创建提供了更大的灵活性和扩展性,是实际开发中使用频率极高的一种设计模式。
一方面,我们将简要回顾 Day 1 的单例模式,帮助建立起模式之间的连接;另一方面,重点学习如何通过工厂方法封装对象创建逻辑,做到"开闭原则"。

一、Day 1 回顾:单例模式
单例模式的核心要点:
- 保证类的唯一实例。
- 提供一个全局访问点。
- 常见实现包括懒汉式、饿汉式、静态内部类、双重检查锁等。
- 推荐 C++11 使用局部静态变量方式实现线程安全单例。
代码示例(局部静态单例):
class Logger {
private:Logger() {}
public:static Logger* getInstance() {static Logger instance;return &instance;}void log(const std::string& msg) {std::cout << "[LOG]: " << msg << std::endl;}
};
二、本日主题:工厂方法模式
1. 意图说明
工厂方法模式:定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类。
核心解决:将对象创建的控制权下放到子类中。
2. 结构与角色
+-------------------+ +---------------------+
| Product |<------+ ConcreteProductA |
+-------------------+ +---------------------+^ ^| |
+--------------+ +------------------+
| Factory |<-------| ConcreteFactoryA |
+--------------+ +------------------+
| + create() | | + create() |
+--------------+ +------------------+
- Product:抽象产品类
- ConcreteProduct:具体产品类
- Factory:抽象工厂类,声明工厂方法
- ConcreteFactory:具体工厂类,负责生产具体产品
3. 示例:日志系统的多种实现
我们希望在程序中根据不同配置,使用不同类型的日志(如控制台日志、文件日志等)。使用工厂方法模式如下:
(1) 抽象产品接口
class Logger {
public:virtual void log(const std::string& msg) = 0;virtual ~Logger() {}
};
(2) 具体产品实现
class ConsoleLogger : public Logger {
public:void log(const std::string& msg) override {std::cout << "[Console] " << msg << std::endl;}
};class FileLogger : public Logger {
public:void log(const std::string& msg) override {// 简化处理,真实应写入文件std::cout << "[File] " << msg << std::endl;}
};
(3) 工厂接口
class LoggerFactory {
public:virtual Logger* createLogger() = 0;virtual ~LoggerFactory() {}
};
(4) 具体工厂类
class ConsoleLoggerFactory : public LoggerFactory {
public:Logger* createLogger() override {return new ConsoleLogger();}
};class FileLoggerFactory : public LoggerFactory {
public:Logger* createLogger() override {return new FileLogger();}
};
(5) 客户端使用
void testFactory(LoggerFactory* factory) {Logger* logger = factory->createLogger();logger->log("This is a test message.");delete logger;
}int main() {ConsoleLoggerFactory consoleFactory;testFactory(&consoleFactory);FileLoggerFactory fileFactory;testFactory(&fileFactory);return 0;
}
4. 优缺点总结
✅ 优点:
- 满足开闭原则:新增产品时只需添加新工厂和产品类,无需改动原有逻辑。
- 屏蔽具体产品类,解耦创建逻辑与使用逻辑。
❌ 缺点:
- 类的数量增多:每种产品都需要对应工厂类。
- 结构相对复杂,不适合产品类型单一的简单场景。
三、适用场景总结
| 场景 | 原因 |
|---|---|
| 日志模块支持多种输出方式 | 控制台、文件、网络输出分开实现 |
| 图像加载库支持多种格式 | 不同格式(PNG/JPEG/GIF)封装为产品类 |
| 数据库驱动连接 | MySQL/PostgreSQL/SQLite 等切换方便 |
四、今日练习题
✍️ 题目一:
设计一个简化的图像解码系统,要求支持解码 PNG 和 JPEG 图像,使用工厂方法模式设计产品与工厂类结构。
✍️ 题目二:
工厂方法模式如何比简单工厂模式更符合开闭原则?请结合实际项目经验说明。
五、滚动复习:Day 1 要点回顾
- 单例模式确保类只有一个实例,常用于日志、配置、线程池等场景;
- 推荐使用 C++11 局部静态变量实现线程安全单例;
- 注意拷贝构造与赋值运算符需禁用;
- 单例不易测试,可能违反开闭原则。
六、小结与展望
今天我们学习了工厂方法模式,掌握了如何将对象创建交给子类工厂完成,以提高系统的灵活性和可扩展性。明日我们将学习抽象工厂模式(Abstract Factory),它是工厂方法模式的进一步扩展,适用于产品族的创建场景。
持续学习,稳步进阶,设计模式每天一点点,收获将是长远的代码质量提升与架构思维成长。
相关文章:
设计模式 Day 2:工厂方法模式(Factory Method Pattern)详解
继 Day 1 学习了单例模式之后,今天我们继续深入对象创建型设计模式——工厂方法模式(Factory Method)。工厂方法模式为对象创建提供了更大的灵活性和扩展性,是实际开发中使用频率极高的一种设计模式。 一方面,我们将简…...
自动驾驶浪潮下,HMI 设计如何保障安全与便捷?
自动驾驶系统与 HMI 设计的关联性 自动驾驶系统涵盖了一系列复杂的传感器技术、算法以及执行机构。从激光雷达、摄像头等环境感知传感器,到用于处理海量数据的人工智能算法,再到控制车辆行驶的动力与转向执行系统,各部分协同工作,…...
瑞昱RTD2556QR显示器驱动芯片
一、概述 RTD2556QR芯片是由Realtek公司精心研发的一款高性能显示驱动芯片,专为满足现代显示设备对高分辨率、多功能接口及稳定性能的需求而设计。该芯片凭借其卓越的技术特性和广泛的应用领域,在显示驱动市场中占据重要地位。它集成了多种先进的功能模…...
复合缩放EfficientNet原理详解
1. 为什么复合缩放更高效? (1)单维度缩放的瓶颈 增加深度(层数): 更深的网络可以学习更复杂特征,但容易导致梯度消失/爆炸问题,且计算量随深度线性增长。 问题:深层网络…...
线程等待与唤醒的几种方法与注意事项
写在前面:无论是调用哪种等待和唤醒的方法,都必须是当前线程所持有的对象,否则会导致 java.lang.IllegalMonitorStateException 等并发安全问题。 以三个线程循环打印 XYZ 为例。 一、方法 1.1 Object 对象锁 可以通过 synchronized 对方…...
rustdesk 客户端使用
配置中继服务器 RustDesk 搭建-CSDN博客 配置客户端,服务端(控制方,被控方) 1.下载rustdesk.exe(windows为例) 2.完成后如下 3.配置...
react+antd封装一个可回车自定义option的select并且与某些内容相互禁用
需求背景 一个select框 现在要求可多选 并且原有一个any的选项 其他选项为输入后回车自己增加 若选择了any 则其他选项不可选择反之选择其他选项any不可选择 并且回车新增时也不可直接加入到选中数组只加入到option内 并且不可重复添加新内容 实现过程 <Form.Item …...
碳化硅 MOSFET三相逆变电路损耗新算法
基 于 碳 化 硅 MOSFET三相逆变电路损耗新算法 摘 要 提出了一种三相逆变电路功率开关器件损耗计算的新方法.为了达到将高频电力电子电路和实时仿真算 法 相 结 合 应 用 于 嵌 入 式 实 时 仿 真 平 台 的 目 的 ,针 对 工 程 应 用 中 逆 变 器 损 耗 计 算 的 实…...
增加等IO状态的唤醒堆栈打印及缺页异常导致iowait分析
一、背景 在之前的博客 在计算进程D状态持续时间及等IO的时间遇到的一处问题-CSDN博客 里,我们修复了一处在抓取D状态及等IO状态堆栈的监控程序的一处时间计算bug,在这篇博客里,我们进一步丰富监控程序,在进程iodelay被唤醒时&am…...
nodejs:midi-writer-js 将基金净值数据转换为 midi 文件
开放式基金是没有公布每日交易量的。 /funds/data/660008.csv 文件开头: date,jz,ljjz 2016-01-04,1.1141,1.1141 2016-01-05,1.1161,1.1161 2016-01-06,1.1350,1.1350 这是一个将开放式基金数据转换为 MIDI音乐的 js 程序示例。该程序将基金净值映射为 MIDI音符的…...
新能源汽车空调系统(R134A)性能评估(一)
国内外主流空调系统厂家:贝尔、德尔福、空调国际、法雷奥、电装、松芝、杰信、新电、豫新等 泛亚汽车的空调电子部是比较优秀的整车空调研发团队。 空调系统综合试验台架是一套由试验室、风量测定装置、空气调和器、空气温度测定装置、湿度测定装置、加热器试验辅助…...
Oracle 数据库中优化 INSERT INTO 操作的性能
在 Oracle 数据库中优化 INSERT INTO 操作的性能,尤其是在处理大批量数据时,可以通过以下方法显著提升效率。 使用直接路径插入(Direct-Path Insert) 通过 APPEND 提示绕过缓冲区缓存,直接写入数据文件,减…...
Ubuntu 22.04安装MongoDB:GLM4模型对话数据收集与微调教程
在Ubuntu 22.04安装MongoDB Community Edition的教程请点击下方链接进行参考: 点击这里获取MongoDB Community Edition安装教程 今天将为大家带来如何微调GLM4模型并连接数据库进行对话的教程。快跟着小编一起试试吧~ 1. 大模型 ChatGLM4 微调步骤 1.1 从 github…...
Java 中的继承与多态:面向对象编程的核心特性
继承和多态是面向对象编程中最重要的两个概念,它们使代码结构更加清晰、灵活,并极大地提高了代码复用性。本文将深入探讨 Java 中的继承与多态,帮助你更好地理解这些核心概念。 1. 继承 1.1 为什么需要继承 在实际编程中,我们经…...
可编程增益放大器(PGA)在智能传感器自调节系统中的角色
在电子电路设计中,放大器芯片作为信号处理的核心器件,其性能直接影响系统整体表现。然而面对运算放大器、功率放大器、仪表放大器等众多类型,工程师常陷入选型困惑。作为国内领先的半导体解决方案提供商,华芯邦深耕放大器芯片领域…...
微信登录、商品浏览前瞻
一.业务效果 二.所需技术...
浙大研究团队揭示电场调控5-HT1AR的分子机制
本期介绍的文章题为 “Structural Insight into the Inactive/Active States of 5‑HT1AR and Molecular Mechanisms of Electric Fields in Modulating 5‑HT1AR” 。近期发表于JCIM。通过分子动力学模拟,探究 5-羟色胺 1A 受体(5-HT1AR) 在非活性 / 活性状态的构象…...
RoboOS与RoboBrain:引领具身智能新时代的跨本体协作框架
摘要 2025年3月29日,智源研究院在中关村论坛的“未来人工智能先锋论坛”上发布了两项重要成果:跨本体具身大小脑协作框架RoboOS与开源具身大脑RoboBrain。这一创新技术使机器人能够实现跨场景多任务部署及跨本体协作,推动单机智能向群体智能…...
视频AI赋能水利行业生态治理,水电站大坝漂浮物实时监测与智能预警方案
水电站大坝周边水域垃圾漂浮物不仅影响水质,还可能对大坝设施运行、水生态环境造成威胁。传统依靠人工巡检的方式效率低、存在监测盲区,难以实时全面地掌握漂浮物情况。借助EasyCVR视频汇聚平台与TSINGSEE青犀AI算法中台构建智能化监测方案,能…...
SnapdragonCamera骁龙相机源码解析
骁龙相机是高通开发的一个测试系统摄像头的demo,代码完善,功能强大。可以配合Camera驱动进行功能联调。 很多逻辑代码在CaptureModule.java里。 CaptureModule有8000多行,包罗万象。 涉及到界面显示要结合CaptureUI.java 一起来实现。 Ca…...
Spring Boot 整合 RabbitMQ:注解声明队列与交换机详解
RabbitMQ 作为一款高性能的消息中间件,在分布式系统中广泛应用。Spring Boot 通过 spring-boot-starter-amqp 提供了对 RabbitMQ 的无缝集成,开发者可以借助注解快速声明队列、交换机及绑定规则,极大简化了配置流程。本文将通过代码示例和原理…...
SecureCRT常用命令
一、连接命令 1. telnet:使用Telnet协议连接到远程设备。 例如:telnet 192.168.1.1 会连接到IP地址为192.168.1.1的远程设备。 二、文件传输命令 2. put:上传文件到远程服务器。 例如:put C:\localfile.txt /remotefolder 将本地文…...
flink 分组窗口聚合 与 窗口表值函数聚合 的区别
警告:分组窗口聚合已经过时。推荐使用更加强大和有效的窗口表值函数聚合。 参考官方文档 在 Apache Flink 中,分组窗口聚合(Group Window Aggregation) 和 窗口表值函数聚合(Windowing TVF Aggregation)…...
阿里云Tair KVCache:打造以缓存为中心的大模型Token超级工厂
一、Tair KVCache 简介 Tair KVCache 是阿里云瑶池旗下云数据库 Tair 面向大语言模型推理场景推出的 KVCache 缓存加速服务。 随着互联网技术的演进与流量规模的激增,缓存技术逐渐成为系统架构的核心组件。该阶段催生了 Redis 等开源缓存数据库,阿里巴巴…...
通过TIM+DMA Burst 实现STM32输出变频且不同脉冲数量的PWM波形
Burst介绍: DMA控制器可以生成单次传输或增量突发传输,传输的节拍数为4、8或16。 为了确保数据一致性,构成突发传输的每组传输都是不可分割的:AHB传输被锁定,AHB总线矩阵的仲裁器在突发传输序列期间不会撤销DMA主设备…...
【JAVA】【疑难杂症解决!】org.springframework.transaction.UnexpectedRollbackException:
程序莫名其妙出现这个事务回滚错误,不显示具体错误信息!! 问题:前几天遇到一个问题,代码没有抛出我想要的带自定义提示消息的异常,却报了个这个,去搜了一下,大概原因如下: 因为我在方法上写了@Transactional注解,里边调用的service的方法上也写了@Transactional注解…...
[Effective C++]条款26:尽可能延后变量定义的出现时间
. 在C中,尽可能延后变量定义的出现时间,主要原因是为了提供代码的可读性,减少不必要的开销以及避免潜在的错误。 1、代码执行过程中抛出异常 如果在代码开头定义了变量,但在后续代码中抛出了异常,可能导致变量在未被使…...
如何在k8s中对接s3存储
github地址: https://github.com/majst01/csi-driver-s3 1.CSI for S3 这是用于 S3(或兼容 S3)存储的容器存储接口 (CSI)。它可以动态分配存储桶并通过Fuse mount将它们安装到任何容器中 2.状态 这仍处于试验阶段,不应在任何…...
基于TradingView和CTPBee的自动化期货交易系统实现
引言 在量化交易领域,TradingView因其强大的技术分析工具和丰富的指标库而广受欢迎,但是其不支持国内期货自动化交易,CTPBee则是一个优秀的国产Python期货交易接口。本文将介绍如何将两者结合,实现一个完整的自动化交易系统。 本…...
FPGA实现LED流水灯
一、在VsCode中写代码 1、建立工程项目文件water_led.v文件 2、打开项目文件,创建三个目录 3、打开文件trl,创建water_led.v文件 4、打开文件tb,创建water_led_tb.v文件 5、用VsCode打开water_led.v文件,编写源代码 module water…...
