【第二节】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 以及前端知识仍然非常必要,而且在可预见的未来,…...
【数据标准】数据标准化是数据治理的基础
导读:数据标准化是数据治理的基石,它通过统一数据格式、编码、命名与语义等,全方位提升数据质量,确保准确性、完整性与一致性,从源头上杜绝错误与冲突。这不仅打破部门及系统间的数据壁垒,极大促进数据共享…...
ES6从入门到精通:前言
ES6简介 ES6(ECMAScript 2015)是JavaScript语言的重大更新,引入了许多新特性,包括语法糖、新数据类型、模块化支持等,显著提升了开发效率和代码可维护性。 核心知识点概览 变量声明 let 和 const 取代 var…...
376. Wiggle Subsequence
376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...
测试markdown--肇兴
day1: 1、去程:7:04 --11:32高铁 高铁右转上售票大厅2楼,穿过候车厅下一楼,上大巴车 ¥10/人 **2、到达:**12点多到达寨子,买门票,美团/抖音:¥78人 3、中饭&a…...
(二)原型模式
原型的功能是将一个已经存在的对象作为源目标,其余对象都是通过这个源目标创建。发挥复制的作用就是原型模式的核心思想。 一、源型模式的定义 原型模式是指第二次创建对象可以通过复制已经存在的原型对象来实现,忽略对象创建过程中的其它细节。 📌 核心特点: 避免重复初…...
基于数字孪生的水厂可视化平台建设:架构与实践
分享大纲: 1、数字孪生水厂可视化平台建设背景 2、数字孪生水厂可视化平台建设架构 3、数字孪生水厂可视化平台建设成效 近几年,数字孪生水厂的建设开展的如火如荼。作为提升水厂管理效率、优化资源的调度手段,基于数字孪生的水厂可视化平台的…...
Nuxt.js 中的路由配置详解
Nuxt.js 通过其内置的路由系统简化了应用的路由配置,使得开发者可以轻松地管理页面导航和 URL 结构。路由配置主要涉及页面组件的组织、动态路由的设置以及路由元信息的配置。 自动路由生成 Nuxt.js 会根据 pages 目录下的文件结构自动生成路由配置。每个文件都会对…...
Psychopy音频的使用
Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...
Spring Boot+Neo4j知识图谱实战:3步搭建智能关系网络!
一、引言 在数据驱动的背景下,知识图谱凭借其高效的信息组织能力,正逐步成为各行业应用的关键技术。本文聚焦 Spring Boot与Neo4j图数据库的技术结合,探讨知识图谱开发的实现细节,帮助读者掌握该技术栈在实际项目中的落地方法。 …...
Rust 异步编程
Rust 异步编程 引言 Rust 是一种系统编程语言,以其高性能、安全性以及零成本抽象而著称。在多核处理器成为主流的今天,异步编程成为了一种提高应用性能、优化资源利用的有效手段。本文将深入探讨 Rust 异步编程的核心概念、常用库以及最佳实践。 异步编程基础 什么是异步…...
c#开发AI模型对话
AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...
