C++设计模式——抽象工厂模式

文章目录

    • 抽象工厂模式的主要组成部分
    • 抽象工厂模式的一个典型例子
    • 抽象工厂模式用于其他场景
    • 抽象工厂模式与其他设计模式结合使用

C++ 中的抽象工厂模式是一种创建型设计模式,它主要用于处理对象家族的创建,这些对象之间可能存在一定的关联关系或属于相同的产品族。抽象工厂模式的核心目标是提供一个接口,允许客户端通过此接口创建一系列相关或相互依赖的对象,而不必知道具体产生的对象的具体类。

抽象工厂模式的主要组成部分

  1. 抽象工厂(Abstract Factory):这是一个接口,声明了一组用于创建相关或依赖对象的方法,每个方法对应一种产品对象。

    class AbstractFactory {
    public:virtual ~AbstractFactory() {}virtual IProductA* createProductA() = 0; // 创建产品A的抽象方法virtual IProductB* createProductB() = 0; // 创建产品B的抽象方法// ... 可能还有其他产品方法
    };
    
  2. 具体工厂(Concrete Factory):这是抽象工厂接口的具体实现类,它负责创建一个产品族内的具体产品对象。

    class ConcreteFactory1 : public AbstractFactory {
    public:IProductA* createProductA() override { return new ConcreteProductA1(); }IProductB* createProductB() override { return new ConcreteProductB1(); }
    };class ConcreteFactory2 : public AbstractFactory {
    // ...
    };
    
  3. 产品接口(Product Interface):代表产品对象的抽象接口,定义了产品的公共方法。

    class IProductA {
    public:virtual ~IProductA() {}// 声明产品A的相关操作
    };class IProductB {
    public:virtual ~IProductB() {}// 声明产品B的相关操作
    };
    
  4. 具体产品(Concrete Product):实现了产品接口的具体产品类。

    class ConcreteProductA1 : public IProductA {
    // 实现产品A1的具体行为
    };class ConcreteProductB1 : public IProductB {
    // 实现产品B1的具体行为
    };// 同样会有 ConcreteProductA2, ConcreteProductB2 等...
    

使用抽象工厂模式的优势在于:

  • 隔离了具体的产品实现:客户端只需关心抽象工厂的接口,无需了解创建对象的具体细节。
  • 方便产品族的整体替换:如果想切换到同一产品族的不同实现,只需更改使用的具体工厂即可。
  • 确保产品一致性:通过抽象工厂创建的对象遵循一套一致的设计规则,适合那些必须一起工作的对象集合。

然而,抽象工厂模式也有其局限性,例如难以添加新的产品种类,因为它要求修改抽象工厂接口以及相应的具体工厂实现。此外,随着产品数量的增加,系统的复杂性也会提高。

抽象工厂模式的一个典型例子

在实际应用中,抽象工厂模式的一个典型例子是操作系统 GUI 库的创建。比如,我们有一个抽象工厂来创建按钮、文本框等各种 GUI 组件,而每个具体工厂(如 WindowsFactory 或 LinuxFactory)则负责创建相应操作系统下的具体 GUI 组件。

// 抽象工厂接口
class GUIFactory {
public:virtual ~GUIFactory() {}virtual Button* createButton() const = 0;virtual TextBox* createTextBox() const = 0;// 其他组件的创建方法...
};// 具体工厂
class WindowsFactory : public GUIFactory {
public:Button* createButton() const override { return new WindowsButton(); }TextBox* createTextBox() const override { return new WindowsTextBox(); }// ...
};class LinuxFactory : public GUIFactory {
public:Button* createButton() const override { return new LinuxButton(); }TextBox* createTextBox() const override { return new LinuxTextBox(); }// ...
};// 产品接口
class Button {
public:virtual ~Button() {}virtual void draw() const = 0;// ...
};class TextBox {
public:virtual ~TextBox() {}virtual void draw() const = 0;// ...
};// 具体产品
class WindowsButton : public Button {
public:void draw() const override { /*绘制Windows风格的按钮*/ }// ...
};class WindowsTextBox : public TextBox {
public:void draw() const override { /*绘制Windows风格的文本框*/ }// ...
};// Linux下对应的Button和TextBox实现...

在客户端代码中,我们可以根据需要选择合适的工厂来创建一整套风格一致的界面组件:

int main() {GUIFactory* factory;if (isWindowsPlatform()) {factory = new WindowsFactory();} else if (isLinuxPlatform()) {factory = new LinuxFactory();}Button* myButton = factory->createButton();TextBox* myTextBox = factory->createTextBox();// 使用创建出来的组件myButton->draw();myTextBox->draw();delete myButton;delete myTextBox;delete factory;return 0;
}

这样,无论操作系统环境如何变化,客户端代码都可以保持不变,体现了开闭原则——对扩展开放,对修改关闭。同时,由同一家工厂创建出来的组件具有内在的一致性,可以很好地协同工作。

抽象工厂模式用于其他场景

此外,抽象工厂模式还常用于其他场景,例如:

  • 数据库访问:抽象工厂可以定义创建不同数据库连接、执行SQL命令等操作的方法,而具体工厂可以是MySQLFactory、OracleFactory等,分别用来创建各自数据库系统的连接对象和查询命令对象。
// 抽象工厂
class DatabaseFactory {
public:virtual ~DatabaseFactory() {}virtual Connection* createConnection() const = 0;virtual Query* createQuery() const = 0;
};// 具体工厂
class MySQLFactory : public DatabaseFactory {
public:Connection* createConnection() const override { return new MySQLConnection(); }Query* createQuery() const override { return new MySQLQuery(); }
};class OracleFactory : public DatabaseFactory {
public:Connection* createConnection() const override { return new OracleConnection(); }Query* createQuery() const override { return new OracleQuery(); }
};// 产品接口
class Connection {
public:virtual ~Connection() {}virtual bool connect(const std::string& host, int port, const std::string& user, const std::string& password) = 0;// 其他连接相关方法...
};class Query {
public:virtual ~Query() {}virtual ResultSet* execute(const std::string& sql) = 0;// 其他查询相关方法...
};// 具体产品类...

通过这种设计模式,软件架构得以解耦,使得各个部分能够独立变化和发展,同时确保了系统内部组件间的兼容性和一致性。在复杂的应用场景中,抽象工厂模式尤其有助于模块化设计和维护。

抽象工厂模式与其他设计模式结合使用

在更高级别的软件体系结构设计中,抽象工厂模式还可以与其他设计模式结合使用,以构建更为灵活和可扩展的解决方案。例如:

  • 与策略模式结合:抽象工厂可以返回一组实现了某种策略接口的对象,这样客户可以根据需求选择不同的策略组合。例如,在图形渲染引擎中,抽象工厂可以创建不同的渲染策略(如光照策略、纹理策略),然后将它们组合起来形成特定的效果。

  • 与工厂方法模式结合:抽象工厂中的某些方法可以进一步采用工厂方法模式,即在具体工厂内定义创建单个产品的工厂方法,使得产品创建逻辑更加灵活且易于扩展。

  • 与装饰器模式配合:当产品家族中存在需要逐步增强或修改的组件时,可以在获取基础组件后,通过装饰器模式为其添加额外的功能或属性。

  • 与服务定位器模式搭配:在大型系统中,抽象工厂可以与服务定位器模式相结合,通过服务定位器查找并注入所需的抽象工厂实例,以便在运行时决定具体使用哪种工厂。

总的来说,抽象工厂模式是一个强大而实用的设计模式,它不仅帮助我们组织和管理相关联的产品族,而且在很大程度上增强了代码的可复用性和可扩展性。但在实际应用时需要注意权衡其带来的灵活性与增加的复杂性,尤其是在产品家族变动频繁或者产品种类较少的情况下,直接使用简单工厂或工厂方法模式可能更为合适。

python推荐学习汇总连接:
50个开发必备的Python经典脚本(1-10)

50个开发必备的Python经典脚本(11-20)

50个开发必备的Python经典脚本(21-30)

50个开发必备的Python经典脚本(31-40)

50个开发必备的Python经典脚本(41-50)
————————————————

​最后我们放松一下眼睛
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/499383.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

开仓济民 SWUST OJ 1175

有个王国在某年爆发了旱灾,于是那里的国王准备开仓济民。已知,第一天发一公斤粮食,第二天发两公斤粮食, 第三天发四公斤粮食。。。。。第二天发的粮食是前一天的两倍。我们已知安抚灾民一共需要m公斤粮食,问第几天国 王…

接口被恶意刷了!老板怒吼一声而我打开这篇文章!

在网络安全领域,恶意刷接口是一种常见的攻击手段,可能导致服务资源耗尽、数据泄露等严重后果。为了应对这一问题,我们需要采取一系列防范措施。 防火墙 数据包过滤与验证:防火墙具备对传入和传出网络的数据包进行深度分析和过滤…

【python】Python Turtle绘制流星雨动画效果【附源码】

在这篇技术博客中,我们将学习如何使用 Python 的 Turtle 模块绘制一个流星雨的动画效果。通过简单的代码实现,我们可以在画布上展现出流星闪耀的场景,为视觉带来一丝神秘与美感。 一、效果图: 二、准备工作 (1)、导入…

uniapp 安装安卓、IOS模拟器并调试

一、安装Android模拟器并调试 1.下载并安装Android Studio。 2.创建简单project。 3.安装模拟器。 完成安卓模拟器的安装。 4.启动模拟器。 5.hbuilderx选择模拟器、运行。 点击刷新按钮后出现模拟器,勾选并运行。 6.调试。 在 HBuilderX 中,项目启…

Aigtek高精度电流源仪器设计规范

高精度电流源仪器是一种用于产生和测量精确电流的设备,广泛应用于电子、通信、自动控制等领域。为了确保仪器的性能和可靠性,设计过程中需要遵循一些规范。 电流源仪器的设计要注重稳定性。稳定性是保证仪器输出电流精度的关键因素。设计过程中应选择高精…

数据结构·栈和队列

1. 栈 1.1 栈的概念及结构 栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一段称为栈顶,另一端称为栈底。 栈中的数据元素遵守 后进先出 LIFO(Last In First Out)的原则,后进来的数…

2023年上半年教师资格证考试《综合素质》(中学)题

1.杨老师经常担心的问题是:“这些内容是学生们需要的吗?”“我这样教,学生能接受吗?”这表明杨老师所处的教师专业发展阶段是(C )。 A虚拟关注阶段 B生存关注阶段 C自我更新关注阶段 D任务关注阶段 14.吴老师教学经验丰富&a…

腾讯云优惠:2024云服务器租用价格表,买前必看

一张表看懂腾讯云服务器租用优惠价格表,一目了然,腾讯云服务器分为轻量应用服务器和云服务器CVM,CPU内存配置从2核2G、2核4G、4核8G、8核16G、4核16G、8核32G、16核32G、16核64等配置可选,公网带宽1M、3M、5M、12M、18M、22M、28M…

2.1_6 线程的实现方式和多线程模型

文章目录 2.1_6 线程的实现方式和多线程模型(一)线程的实现方式(1)用户级线程(2)内核级线程 (二)多线程模型(1)一对一模型(2)多对一模…

内核打印应用程序出错信息,DEBUG_USER

前言 在 Linux 系统中,运行一个应用程序,突然提示段错误,并停止运行 # ./crash.out Segmentation fault如果这个时候操作系统能多提示点错误信息,那将会缩短我们 debug 的时间。 core dump 就是一个办法,可以查看我…

C# EF Core迁移数据库

现象: 在CodeFirst时,先写字段与表,创建数据库后,再添加内容 但字段与表会变更,比如改名删除增加等 需求: 当表字段变更时,同时变更数据库,执行数据库迁移 核心命令 Add-Migrat…

06|Mysql内部组件结构

1. 连接器 客户端要向mysql发起通信都必须先跟Server端建立通信连接,而建立连接的工作就是由连接器完成的 mysql -h host[数据库地址] -u root[用户] -p root[密码] -P 3306连接步骤: 1、如果用户名或密码不对,你就会收到一个"Access denied for us…