【第二节】C++设计模式(创建型模式)-抽象工厂模式
目录
引言
一、抽象工厂模式概述
二、抽象工厂模式的应用
三、抽象工厂模式的适用场景
四、抽象工厂模式的优缺点
五、总结
引言
抽象工厂设计模式是一种创建型设计模式,旨在解决一系列相互依赖对象的创建问题。它与工厂方法模式密切相关,但在应用场景和实现方式上有显著区别。本文将通过理论讲解和代码示例,深入探讨抽象工厂模式的核心思想、适用场景及其优缺点。
一、抽象工厂模式概述
(1) 核心思想
抽象工厂模式通过定义一个创建一系列相关或相互依赖对象的接口,将具体对象的创建工作封装到具体的工厂类中。它强调面向接口编程,支持多种变化需求,同时确保生成的对象的关联性和兼容性。
(2)与工厂方法模式的关系
工厂方法模式:解决单一对象的创建问题,通过子类化实现对象的创建。
抽象工厂模式:解决一系列相互依赖对象的创建问题,通过一个工厂类创建多个相关对象。
(3)主要解决的问题
避免直接使用 `new` 操作符导致的对象绑定问题。
支持多种变化需求(如多种数据库、多种产品系列)。
确保生成的对象的关联性和兼容性。
(4)数据库访问层示例
需求分析
在数据库访问层中,通常需要创建以下对象:
数据库连接(`IDbConnection`)
命令对象(`IDbCommand`)
数据读取器对象(`IDataReader`)
不同数据库(如 MySQL、SQL Server)需要不同的实现类。如果直接使用 `new` 关键字创建对象,会导致代码与具体数据库绑定,难以支持多种数据库。
面向接口编程
通过定义接口 `IDbConnection`、`IDbCommand` 和 `IDataReader`,实现面向接口编程。具体数据库的连接、命令和读取器类继承自这些接口,从而支持多种数据库的实现。
工厂方法模式的局限性
工厂方法模式可以解决单一对象的创建问题,但不适用于创建多个相互依赖的对象。例如,不同工厂可能创建不兼容的对象组合,导致错误。
二、抽象工厂模式的应用
(1)解决对象组合问题
抽象工厂模式将多个相互依赖对象的创建工作合并到一个工厂类中。通过一个工厂类创建所有相关对象,确保对象之间的兼容性。
(2)代码示例
以下是一个简单的抽象工厂模式实现示例,展示了如何创建一组相关对象。
Abstract Factory模式的核心价值在于通过抽象工厂接口将一组相关或依赖对象的创建过程统一封装到具体工厂类(ConcreteFactory)中。这种集中化的对象创建机制不仅符合单一职责原则,更重要的是通过消除分散的创建逻辑,显著降低了系统在对象实例化层面的维护复杂度,为产品族的扩展提供了规范的框架基础。
产品类定义
// Product.h
#ifndef _PRODUCT_H_
#define _PRODUCT_H_class AbstractProductA {
public:virtual ~AbstractProductA();
protected:AbstractProductA();
};class AbstractProductB {
public:virtual ~AbstractProductB();
protected:AbstractProductB();
};class ProductA1 : public AbstractProductA {
public:ProductA1();~ProductA1();
};class ProductA2 : public AbstractProductA {
public:ProductA2();~ProductA2();
};class ProductB1 : public AbstractProductB {
public:ProductB1();~ProductB1();
};class ProductB2 : public AbstractProductB {
public:ProductB2();~ProductB2();
};#endif // ~_PRODUCT_H_
工厂类定义
// AbstractFactory.h
#ifndef _ABSTRACTFACTORY_H_
#define _ABSTRACTFACTORY_H_class AbstractProductA;
class AbstractProductB;class AbstractFactory {
public:virtual ~AbstractFactory();virtual AbstractProductA* CreateProductA() = 0;virtual AbstractProductB* CreateProductB() = 0;
protected:AbstractFactory();
};class ConcreteFactory1 : public AbstractFactory {
public:ConcreteFactory1();~ConcreteFactory1();AbstractProductA* CreateProductA();AbstractProductB* CreateProductB();
};class ConcreteFactory2 : public AbstractFactory {
public:ConcreteFactory2();~ConcreteFactory2();AbstractProductA* CreateProductA();AbstractProductB* CreateProductB();
};#endif // ~_ABSTRACTFACTORY_H_
工厂类实现
// AbstractFactory.cpp
#include "AbstractFactory.h"
#include "Product.h"
#include <iostream>
using namespace std;AbstractFactory::AbstractFactory() {}
AbstractFactory::~AbstractFactory() {}ConcreteFactory1::ConcreteFactory1() {}
ConcreteFactory1::~ConcreteFactory1() {}
AbstractProductA* ConcreteFactory1::CreateProductA() {return new ProductA1();
}
AbstractProductB* ConcreteFactory1::CreateProductB() {return new ProductB1();
}ConcreteFactory2::ConcreteFactory2() {}
ConcreteFactory2::~ConcreteFactory2() {}
AbstractProductA* ConcreteFactory2::CreateProductA() {return new ProductA2();
}
AbstractProductB* ConcreteFactory2::CreateProductB() {return new ProductB2();
}
测试程序
// main.cpp
#include "AbstractFactory.h"
#include <iostream>
using namespace std;int main(int argc, char* argv[]) {AbstractFactory* cf1 = new ConcreteFactory1();cf1->CreateProductA();cf1->CreateProductB();AbstractFactory* cf2 = new ConcreteFactory2();cf2->CreateProductA();cf2->CreateProductB();return 0;
}
从实现层面来看,Abstract Factory模式通过具体工厂类(ConcreteFactory1)对产品对象的创建过程进行封装。在测试用例中可见,当需要构建一组相关联的产品对象(如ProductA1、ProductA2)时,客户端仅需依赖统一的工厂接口进行操作,而无需关注具体产品的实例化细节。这种设计不仅降低了代码的耦合度,更重要的是将原本分散在各处的对象创建逻辑集中管理,显著提升了系统的可维护性和扩展性。
在理解设计模式的过程中,区分Abstract Factory模式与Factory模式的概念差异是一个关键的学习点。本质上,Abstract Factory模式旨在为一系列具有关联性或依赖关系的对象族提供统一的创建接口,它关注的是产品家族的创建逻辑;而Factory模式则专注于单一类型对象的实例化过程,通过将对象的创建延迟到子类来实现扩展性,其核心是单一产品的创建机制。从实现角度来看,Abstract Factory模式通常以Factory模式为基础,通过具体工厂类(ConcreteFactory1)来实现对多个关联产品的协同创建,这种层次化的设计体现了模式间的组合应用。
三、抽象工厂模式的适用场景
(1)一系列相互依赖对象的创建
抽象工厂模式适用于需要创建一组相关对象的场景,例如:
数据库访问层中的连接、命令和读取器对象。
游戏开发中的不同等级怪物及其相关对象。
(2)支持多系列对象的需求变化
当需求变化时,抽象工厂模式可以轻松支持更多系列对象的创建工作,而无需修改现有代码。
四、抽象工厂模式的优缺点
优点
高内聚性:将相关对象的创建集中在一个工厂类中,提高代码的内聚性。
易于扩展:支持新增系列对象的创建,符合开闭原则。
降低耦合:通过面向接口编程,降低客户端与具体实现类的耦合。
缺点
难以应对新对象的需求变动:如果需要新增对象类型,可能需要修改抽象工厂接口及其所有实现类。
复杂性增加:随着系列对象的增多,工厂类的数量也会增加,导致系统复杂性上升。
五、总结
抽象工厂模式是一种强大的设计模式,适用于解决一系列相互依赖对象的创建问题。它通过面向接口编程和封装对象创建逻辑,提高了代码的灵活性和可维护性。然而,在面对新对象需求变动时,抽象工厂模式可能显得不够灵活。因此,在实际开发中,应根据具体需求选择合适的设计模式。
相关文章:

【第二节】C++设计模式(创建型模式)-抽象工厂模式
目录 引言 一、抽象工厂模式概述 二、抽象工厂模式的应用 三、抽象工厂模式的适用场景 四、抽象工厂模式的优缺点 五、总结 引言 抽象工厂设计模式是一种创建型设计模式,旨在解决一系列相互依赖对象的创建问题。它与工厂方法模式密切相关,但在应用…...

【学习资料】嵌入式人工智能Embedded AI
图片来源: Embedded Artificial Intelligence for Business Purposes | DAC.digital 随着AI在设备端的应用,我们看到越来越多的可穿戴设备出现以及自动驾驶汽车的发展,可以看到嵌入式人工智能是新的发展方向。我为大家介绍嵌入式人工智能的…...
【Python爬虫(60)】解锁社交媒体数据宝藏:Python爬虫实战攻略
【Python爬虫】专栏简介:本专栏是 Python 爬虫领域的集大成之作,共 100 章节。从 Python 基础语法、爬虫入门知识讲起,深入探讨反爬虫、多线程、分布式等进阶技术。以大量实例为支撑,覆盖网页、图片、音频等各类数据爬取ÿ…...

C++ 继承,多态
看前须知: 本篇博客是作者听课时的笔记,不喜勿喷,若有疑问可以评论区一起讨论。 继承 定义: 继承机制是⾯向对象程序设计使代码可以复⽤的最重要的⼿段,它允许我们在保持原有 类特性的基础上进⾏扩展,增…...
Java中的Stream API:从入门到实战
引言 在现代Java开发中,Stream API 是处理集合数据的强大工具。它不仅让代码更加简洁易读,还能通过并行处理提升性能。本文将带你从基础概念入手,逐步深入Stream API的使用,并通过实战案例展示其强大功能。 1. 什么是Stream API…...

QPainter绘制3D 饼状图
先展示图片 核心代码如下: pie.h #ifndef Q3DPIE_H #define Q3DPIE_H#include <QtGui/QPen> #include <QtGui/QBrush>class Pie { public:double value; QBrush brush; QString description; double percentValue;QString p…...
【FAQ】HarmonyOS SDK 闭源开放能力 —Live View Kit (1)
1.问题描述: 客户端创建实况窗后,通过Push kit更新实况窗内容,这个过程是自动更新的还是客户端解析push消息数据后填充数据更新?客户端除了接入Push kit和创建实况窗还需要做什么工作? 解决方案: 通过Pu…...
数据治理与管理
引入 上一篇我们聊了数仓架构设计,它是企业构建数据中台的基石。其本质就是构建一个可靠易用的架构,可以借此将原始数据汇聚、处理,最终转换成可消费使用的数据资源。 在拥有数据资源以后,我们就需要考虑如何利用它,为企业创造价值,让它变成企业的资产而不是负担。也就…...
什么是HTTP/2协议?NGINX如何支持HTTP/2并提升网站性能?
HTTP/2是一种用于在Web浏览器和服务器之间进行通信的协议,旨在提高网站性能和加载速度。它是HTTP/1.1的继任者,引入了许多优化和改进,以适应现代Web应用的需求。HTTP/2的主要目标是减少延迟、提高效率,以及更好地支持并发请求。 …...
安全运维,等保测试常见解决问题。
1. 未配置口令复杂度策略。 # 配置密码安全策略 # vi /etc/pam.d/system-auth # local_users_only 只允许本机用户。 # retry 3 最多重复尝试3次。 # minlen12 最小长度为12个字符。 # dcredit-1 至少需要1个数字字符。 # ucredit-1 至少需要1个大…...

jmeter接口测试(二)
一、不同参数类型的接口测试 二、动态参数接口处理 随机数 工具——>函数助手对话框(Random 1000-10000之间的随机数 变量名为rdn)如下图所示 把上图生成的函数字符串复制到想要使用的地方如下图 三、断言 1、状态断言,200 不能证明…...

Keil ARM Complier Missing Compiler Version 5
使用Keil软件时出现了编译时报错,找不到对应的ARM版本,报错Target Target 1 uses ARM-Compiler Default Compiler Version 5 which is not available. *** Please review the installed ARM Compiler Versions: Manage Project Items - Folders/Extensions to manage ARM Compi…...

【僵尸进程】
【僵尸进程】 目录:知识点1. 僵尸进程的定义2. 僵尸进程产生的原因3. 僵尸进程的危害4. 如何避免僵尸进程 代码示例产生僵尸进程的代码示例避免僵尸进程的代码示例(父进程主动回收)避免僵尸进程的代码示例(信号处理) 运…...

【框架】参考 Spring Security 安全框架设计出,轻量化高可扩展的身份认证与授权架构
关键字:AOP、JWT、自定义注解、责任链模式 一、Spring Security Spring Security 想必大家并不陌生,是 Spring 家族里的一个安全框架,特别完善,但学习成本比较大,不少开发者都觉得,这个框架“很重” 他的…...

【Git 学习笔记_27】DIY 实战篇:利用 DeepSeek 实现 GitHub 的 GPG 密钥创建与配置
文章目录 1 前言2 准备工作3 具体配置过程3.1. 本地生成 GPG 密钥3.2. 导出 GPG 密钥3.3. 将密钥配置到 Git 中3.4. 测试提交 4 问题排查记录5 小结与复盘 1 前言 昨天在更新我的第二个 Vim 专栏《Mastering Vim (2nd Ed.)》时遇到一个经典的 Git 操作问题:如何在 …...
微信小程序地图map全方位解析
微信小程序地图map全方位解析 微信小程序的 <map> 组件是一个功能强大的工具,可以实现地图展示、定位、标注、路径规划等多种功能。以下是全方位解析微信小程序地图组件的知识点: 一、地图组件基础 1. 引入 <map> 组件 在页面的 .wxml 文…...
调试无痛入手
在调试过程中,Step In、Step Over 和 Step Out 是控制代码执行流程的常用操作,帮助开发者逐行或逐块检查代码行为。以下是它们的详细介绍及使用方法: 1. Step In 功能:进入当前行的函数或方法内部,逐行执行其代码。使…...
【蓝桥杯集训·每日一题2025】 AcWing 6135. 奶牛体检 python
6135. 奶牛体检 Week 1 2月21日 农夫约翰的 N N N 头奶牛站成一行,奶牛 1 1 1 在队伍的最前面,奶牛 N N N 在队伍的最后面。 农夫约翰的奶牛也有许多不同的品种。 他用从 1 1 1 到 N N N 的整数来表示每一品种。 队伍从前到后第 i i i 头奶牛的…...
AI发展迅速,是否还有学习前端的必要性?
今天有个小伙伴跟我讨论:“现在 AI 发展迅速,是否还有学习 JS 或者 TS 及前端知识的必要?” 我非常肯定地说: 是的,学习 JavaScript/TypeScript 以及前端知识仍然非常必要,而且在可预见的未来,…...

【数据标准】数据标准化是数据治理的基础
导读:数据标准化是数据治理的基石,它通过统一数据格式、编码、命名与语义等,全方位提升数据质量,确保准确性、完整性与一致性,从源头上杜绝错误与冲突。这不仅打破部门及系统间的数据壁垒,极大促进数据共享…...

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)
题目:3442. 奇偶频次间的最大差值 I 思路 :哈希,时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况,哈希表这里用数组即可实现。 C版本: class Solution { public:int maxDifference(string s) {int a[26]…...
【杂谈】-递归进化:人工智能的自我改进与监管挑战
递归进化:人工智能的自我改进与监管挑战 文章目录 递归进化:人工智能的自我改进与监管挑战1、自我改进型人工智能的崛起2、人工智能如何挑战人类监管?3、确保人工智能受控的策略4、人类在人工智能发展中的角色5、平衡自主性与控制力6、总结与…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件
今天呢,博主的学习进度也是步入了Java Mybatis 框架,目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学,希望能对大家有所帮助,也特别欢迎大家指点不足之处,小生很乐意接受正确的建议&…...

ESP32读取DHT11温湿度数据
芯片:ESP32 环境:Arduino 一、安装DHT11传感器库 红框的库,别安装错了 二、代码 注意,DATA口要连接在D15上 #include "DHT.h" // 包含DHT库#define DHTPIN 15 // 定义DHT11数据引脚连接到ESP32的GPIO15 #define D…...
leetcodeSQL解题:3564. 季节性销售分析
leetcodeSQL解题:3564. 季节性销售分析 题目: 表:sales ---------------------- | Column Name | Type | ---------------------- | sale_id | int | | product_id | int | | sale_date | date | | quantity | int | | price | decimal | -…...

(转)什么是DockerCompose?它有什么作用?
一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用,而无需手动一个个创建和运行容器。 Compose文件是一个文本文件,通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...
Java毕业设计:WML信息查询与后端信息发布系统开发
JAVAWML信息查询与后端信息发布系统实现 一、系统概述 本系统基于Java和WML(无线标记语言)技术开发,实现了移动设备上的信息查询与后端信息发布功能。系统采用B/S架构,服务器端使用Java Servlet处理请求,数据库采用MySQL存储信息࿰…...

【C++特殊工具与技术】优化内存分配(一):C++中的内存分配
目录 一、C 内存的基本概念 1.1 内存的物理与逻辑结构 1.2 C 程序的内存区域划分 二、栈内存分配 2.1 栈内存的特点 2.2 栈内存分配示例 三、堆内存分配 3.1 new和delete操作符 4.2 内存泄漏与悬空指针问题 4.3 new和delete的重载 四、智能指针…...

基于IDIG-GAN的小样本电机轴承故障诊断
目录 🔍 核心问题 一、IDIG-GAN模型原理 1. 整体架构 2. 核心创新点 (1) 梯度归一化(Gradient Normalization) (2) 判别器梯度间隙正则化(Discriminator Gradient Gap Regularization) (3) 自注意力机制(Self-Attention) 3. 完整损失函数 二…...