【设计模式】Head First 设计模式——抽象工厂模式 C++实现

设计模式最大的作用就是在变化和稳定中间寻找隔离点,然后分离它们,从而管理变化。将变化像小兔子一样关到笼子里,让它在笼子里随便跳,而不至于跳出来把你整个房间给污染掉。

设计思想

提供一个接口,让该接口负责创建一系列“相关或者相互依赖的对象”,无需指定他们具体的类。

动机

在软件系统中,经常面临着“一系列相互依赖的对象”的创建工作;同时,由于需求的变化,往往存在更多系列对象的创建工作。

如何应对这种变化?如何绕过常规的对象创建方法(new),提供一种“封装机制”来避免客户程序和这种“多系列具体对象创建工作”的紧耦合?

如果没有应对“多系列对象构建”的需求变化,则没有必要使用Abstract Factory模式,这时候使用简单的工厂完全可以。

“系列对象”指的是在某一特定系列下的对象之间有相互依赖或作用关系。不同系列的对象之间不能相互依赖。

Abstract Factory模式主要在于应对“新系列”的需求变动。其缺点在于难以应对“新对象”的需求变动。

抽象工厂和工厂方法非常相像,区别就在于抽象工厂需要一系列对应的对象,而工厂方法就是唯一对象,因此工厂方法也可以理解成抽象工厂的一个特例。

业务场景

要对 SQL Server 数据库进行操作,需要有 Connection,Command,DataReader等一系列配套的操作。而当需要更换数据库的时候,也是这套操作

如何提高代码复用性

一个非常直观的思路是:

class EmployeeDAO {
public:vector<EmployeeDO> GetEmployees() {SqlConnection* connection = new SqlConnection();connection->ConnectionString("...");SqlCommand* command = new SqlCommand();command->CommandText("...");command->SetConnection(connection);SqlDataReader* reader = command->ExecuteReader();while (reader->Read()) {}}
};

看见这些熟悉的代码:

 SqlConnection* connection = new SqlConnection();SqlCommand* command = new SqlCommand();

如果你看过前面的工厂方法模式,你会很自然的联想到那个,然后将代码改进成这样:

// 数据库访问有关的基类
class IDBConnection {};
class IDBConnectionFactory {
public:virtual IDBConnection* CreateDBConnection() = 0;
};class IDBCommand {};
class IDBCommandFactory {
public:virtual IDBCommand* CreateDBCommand() = 0;
};class IDataReader {};
class IDataReaderFactory {
public:virtual IDataReader* CreateDataReader() = 0;
};// 支持SQL Server
class SqlConnection : public IDBConnection {};
class SqlConnectionFactory : public IDBConnectionFactory {};class SqlCommand : public IDBCommand {};
class SqlCommandFactory : public IDBCommandFactory {};class SqlDataReader : public IDataReader {};
class SqlDataReaderFactory : public IDataReaderFactory {};// 支持Oracle
class OracleConnection : public IDBConnection {};
class OracleConnectionFactory : public IDBConnectionFactory {};class OracleCommand : public IDBCommand {};
class OracleCommandFactory : public IDBCommandFactory {};class OracleDataReader : public IDataReader {};
class OracleDataReaderFactory : public IDataReaderFactory {};class EmployeeDAO {IDBConnectionFactory* dbConnectionFactory;IDBCommandFactory* dbCommandFactory;IDataReaderFactory* dataReaderFactory;public:vector<EmployeeDO> GetEmployees() {IDBConnection* connection = dbConnectionFactory->CreateDBConnection();connection->ConnectionString("...");IDBCommand* command = dbCommandFactory->CreateDBCommand();command->CommandText("...");command->SetConnection(connection);  // 关联性IDBDataReader* reader = command->ExecuteReader();  // 关联性while (reader->Read()) {}}
};

有没有解决问题呢?确实解决了,但是如果细心一点,你会发现:这三个工厂实例化出来的对象应该是一套的:SQL只能用SQL的connection,command以及dataReader,MYSQL只能用MYSQL的,也就是说在构造EmployeeDAO的时候,虽然需要传入三个工厂对象:dbConnectionFactory,dbCommandFactory,dataReaderFactory,但是这三个对象却必须是一套的,这就带来了问题:1. 用户有可能传错对象;2.既然必须是一套的,那么完全可以将其封装成一个对象传进来

于是,便有了抽象工厂模式:将三个配套的操作再封装成一个类,避免产生配套错误

代码案例

// 数据库访问有关的基类
class IDBConnection {};class IDBCommand {};class IDataReader {};// 三个操作,绑定到一起
// 此处是这个模式的稳定部分
class IDBFactory {
public:virtual IDBConnection* CreateDBConnection() = 0;virtual IDBCommand* CreateDBCommand() = 0;virtual IDataReader* CreateDataReader() = 0;
};// 支持SQL Server
class SqlConnection : public IDBConnection {};
class SqlCommand : public IDBCommand {};
class SqlDataReader : public IDataReader {};class SqlDBFactory : public IDBFactory {
public:virtual IDBConnection* CreateDBConnection() = 0;virtual IDBCommand* CreateDBCommand() = 0;virtual IDataReader* CreateDataReader() = 0;
};// 支持Oracle
class OracleConnection : public IDBConnection {};
class OracleCommand : public IDBCommand {};
class OracleDataReader : public IDataReader {};class OracleDBFactory : public IDBFactory {
public:virtual IDBConnection* CreateDBConnection() = 0;virtual IDBCommand* CreateDBCommand() = 0;virtual IDataReader* CreateDataReader() = 0;
};class EmployeeDAO {// 保证是同一个工厂// 是一个 familyIDBFactory* dbFactory;public:vector<EmployeeDO> GetEmployees() {IDBConnection* connection = dbFactory->CreateDBConnection();connection->ConnectionString("...");IDBCommand* command = dbFactory->CreateDBCommand();command->CommandText("...");command->SetConnection(connection);  // 关联性IDBDataReader* reader = command->ExecuteReader();  // 关联性while (reader->Read()) {}}
};

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

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

相关文章

js获得相对路径文件,并上传到服务器

如何通过js获得相对路径文件 已知一个相对路径文件&#xff0c;如何使用js将该文件读取为File格式&#xff0c;最后上传到服务器中呢。 1.最简单的解决方案——fetch 代码 import ./index.scss// js通过相对路径获取文件 function FetchGetLocalFile() {const fetchLocalFile …

centos7升级openssh版本

linux升级openssh版本&#xff0c;升级到8.6p1 小白教程&#xff0c;一看就会&#xff0c;一做就成。 1.下载rpm包 2.编写一键安装脚本&#xff08;然后执行&#xff09; #把所有的rpm包&#xff0c;我都放到了/ydy目录&#xff0c;下面安装时&#xff0c;也指定了这个目录 #编…

mysql、MHA高可用配置即故障切换

MHA概述 一套优秀的MySQL高可用环境下故障切换和主从复制的软件 MHA的出现就是解决MySQL 单点的问题 MySQL故障过程中&#xff0c;MHA能做到0-30秒内自动完成故障切换 MHA能在故障切换的过程中最大程度上保证数据的一致性以达到真正意义上的高可用 MHA的组成&#xff08;核…

助力网络管理的利器:企业办公网络中的VLAN划分策略

企业办公网络的性能和安全性对员工的高效工作和信息安全具有重要意义。在实现这一目标时&#xff0c;VLAN&#xff08;Virtual Local Area Network&#xff09;划分在网络设计中发挥着至关重要的作用。通过将办公网络划分为多个虚拟局域网&#xff0c;VLAN划分可以实现网络资源…

uniapp - 倒计时组件-优化循环时间倒计时

使用定时器的规避方法 为了避免定时器误差导致倒计时计算错误&#xff0c;可以采用一些规避方法&#xff0c;比如将倒计时被中断时的剩余时间记录下来&#xff0c;重新开启定时器时再将这个剩余时间加到新的计算中。同时&#xff0c;为了避免定时器延迟&#xff0c;可以在每次执…

记一次Nginx代理Mysql服务的经历

背景&#xff1a; 根据组长背景描述&#xff0c;具备以下前提 1. Mysql服务器为 某A云厂商的RDS SAAS服务&#xff0c;但是不开通外网服务 2. EC2 服务器一台&#xff0c;某A云厂商LaaS服务&#xff0c;也不开通外网 3.阿里云服务器一台&#xff0c;这台服务器有服务需要连…

Web3面试最常见的10个问题

1、什么是web3&#xff1f;web2和web3的区别&#xff1f; web3是互联网的下一代形态,与当前主流的web2形态有着明显的区别: web2是中心化的,web3是去中心化的。 web2的平台由中心化机构控制,如Facebook、微信等。用户需要通过这些中心化平台才能进行互动。而web3基于区块链去…

5700A福禄克FLUKE 5700A多功能校准器

181/2461/8938Fluke 5700A/5720A 高精度多功能校准器 5700A: 世界级标准产品 通过不断的改进、提高&#xff0c;5700A已经演变为5700A系列II。这是福禄克公司生产的、经过大量测试证明、极为可靠的、高精度校准器。5700A已经在全世界的范围被看作是校准器的标准&#xff0c;具有…

直方图统计增强方法

直方图统计增强方法的原理&#xff1a;   直方图统计增强是一种基于像素值分布的图像增强技术&#xff0c;通过调整像素值的分布来增强图像的对比度和细节。其原理是根据图像的直方图信息&#xff0c;将原始像素值映射到一个新的像素值域&#xff0c;从而改变图像的亮度和对比…

用python开发一个炸金花小游戏

众所周知&#xfeff;扑克牌可谓是居家旅行、桌面交友的必备道具&#xff0c; 今天我们用 Python 来实现一个类似炸金花的扑克牌小游戏&#xff0c;先来看一下基本的游戏规则。 炸&#xff08;诈&#xff09;金花又叫三张牌&#xff0c;是在全国广泛流传的一种民间多人纸牌游戏…

【算法与数据结构】106、LeetCode从中序与后序遍历序列构造二叉树

文章目录 一、题目二、解法三、完整代码 所有的LeetCode题解索引&#xff0c;可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、解法 思路分析&#xff1a;首先我们要知道后序遍历数组的最后一个元素必然是根节点&#xff0c;然后根据根节点在中序遍历数组中的…

postgres源码解析55 Brin Index--2(brinbuild流程)

上一篇讲解了brin index的基本概念以及页布局postgres源码解析54 Brin Index–1&#xff0c;后续会从源码角度对索引的构建、维护等方面进行深入讲解。 1 关键数据结构 2 brinbuild执行流程图 3 brinbuild 函数详解 1 首先调用brin_matepage_init初始化brin meta元数据页&…

【0904作业】QT 完成登陆界面跳转到聊天室+完成学生管理系统的查找和删除功能

一、完成登陆界面跳转到聊天室 1> 项目结构 2> 源码 ① .pro ②main #include "mywnd.h" #include"chatCli.h" #include <QApplication>int main(int argc, char *argv[]) {QApplication a(argc, argv);MyWnd w;w.show();Form f;QObject::co…

Lite transformer

图片以及思想来源请参考论文 Lite Transformer with Long-Short Range Attention 瓶颈结构&#xff08;bottleneck&#xff09;是否真的有效 注意力机制被广泛应用在诸多领域&#xff0c;包括自然语言处理&#xff0c;图像处理和视频处理。它通过计算所有输入元素的点积来建模…

低代码平台:IVX 重新定义编程

目录 &#x1f36c;一、写在前面 &#x1f36c;二、低代码平台是什么 &#x1f36c;三、为什么程序员和技术管理者不太可能接受“低代码”平台&#xff1f; &#x1f36d;1、不安全&#xff08;锁定特性&#xff09; &#x1f36d;2、不信任 &#x1f36c;四、IVX低代码平台 &a…

《阿里巴巴 Java 开发手册》(一)命名风格

《阿里巴巴 Java 开发手册》 一、编程规约(一)命名风格(二)常量定义(三)代码格式 一、编程规约 (一)命名风格 (二)常量定义 (三)代码格式

javaScript:DOM(父子/兄弟)常用属性

目录 前言 一.父子关系 父子关系的常用属性 childNodes 获取所有的子节点 children 获取所有的子元素&#xff08;dom元素&#xff09; firstChild 获取元素的第一个子节点&#xff0c;相当于 childNodes[0] firstElementChild 获取元素的第一个元素 相当于 children[0]…

Java是如何实现线程间通信的?

在Java中&#xff0c;线程间通信可以通过以下方式实现&#xff1a; 1.共享变量 线程可以通过共享的变量进行通信。多个线程可以读写同一个变量来交换信息。在这种情况下&#xff0c;需要确保线程对共享变量的访问是同步的&#xff0c;以避免数据竞争和不一致的结果。 以下是一…

spring service事务传播

spring定义的事务行为有以下几种&#xff1a; REQUIRED--支持当前事务&#xff0c;如果当前没有事务&#xff0c;就新建一个事务。这是最常见的选择。 SUPPORTS--支持当前事务&#xff0c;如果当前没有事务&#xff0c;就以非事务方式执行。 MANDATORY--支持当前事务&#xff…

LabVIEW开发感应电机在线匝间短路故障诊断系统

LabVIEW开发感应电机在线匝间短路故障诊断系统 工业中使用的超过85%的电动机是三相感应电动机。它们因其可靠性、设计便利性、高性能和过载能力而被广泛用于不同的应用&#xff0c;例如制造、加工、电力系统、运输等。无论它们的能力如何&#xff0c;它们都被认为是现代工业学…