抽象工厂模式(大话设计模式)C/C++版本
抽象工厂模式

C++
参考:https://www.cnblogs.com/Galesaur-wcy/p/15927110.html
#include <iostream>
using namespace std;// 抽象产品Department ,定义具体产品的公共接口
class Department
{
public:virtual ~Department() = default;virtual void Insert() = 0;virtual Department *GetDepartment() = 0;
};// 创建具体产品类SqlserDepartment, 定义生产的具体产品;
class SqlserDepartment : public Department
{
public:void Insert() override{cout << "在SQL Server 中给 Department 表增加一条记录!" << endl;}Department *GetDepartment() override{cout << "在SQL Server 中根据ID 得到 Department 表一条记录!" << endl;return nullptr;}
};// 创建具体产品类AccessDepartment, 定义生产的具体产品;
class AccessDepartment : public Department
{
public:void Insert() override{cout << "在 Access 中给 Department 表增加一条记录!" << endl;}Department *GetDepartment() override{cout << "在 Access 中根据ID 得到 Department 表一条记录!" << endl;return nullptr;}
};// 抽象产品User ,定义具体产品的公共接口
class User
{
public:virtual ~User() = default;virtual void Insert() = 0;virtual User *GetUser() = 0;
};// 创建具体产品类SqlserverUser, 定义生产的具体产品;
class SqlserverUser : public User
{
public:void Insert() override{cout << "在SQL Server 中给 User表增加一条记录!" << endl;}User *GetUser() override{cout << "在SQL Server 中根据ID 得到 User表一条记录!" << endl;return nullptr;}
};// 创建具体产品类AccessUser, 定义生产的具体产品;
class AccessUser : public User
{
public:void Insert() override{cout << "在 Access 中给 User表增加一条记录!" << endl;}User *GetUser() override{cout << "在 Access 中根据ID 得到 User表一条记录!" << endl;return nullptr;}
};// 抽象工厂,定义具体工厂的公共接口
class AbstractFactory
{
public:virtual ~AbstractFactory() = default;virtual User *CreateUser() = 0;virtual Department *CreateDepartment() = 0;
};// 创建具体工厂SqlServerFactory,定义创建对应具体产品实例的方法;
class SqlServerFactory : public AbstractFactory
{
public:User *CreateUser() override{return new SqlserverUser(); // 该工厂 生产的产品}Department *CreateDepartment() override{return new SqlserDepartment;}
};// 创建具体工厂AccessFactory,定义创建对应具体产品实例的方法;
class AccessFactory : public AbstractFactory
{
public:User *CreateUser() override{return new AccessUser(); // 该工厂 生产的产品}Department *CreateDepartment() override{return new AccessDepartment;}
};int main()
{// AbstractFactory *factory = new SqlServerFactory;AbstractFactory *factory = new AccessFactory;// 调用的是 AccessFactory 工厂,准备生产 SqlserverUser 产品User *iu = factory->CreateUser();iu->Insert();iu->GetUser();// 调用的是 AccessFactory 工厂,准备生产 AccessDepartment 产品Department *id = factory->CreateDepartment();id->Insert();id->GetDepartment();delete factory;delete iu;delete id;factory = nullptr;id = nullptr;id = nullptr;return 0;
}
C
#include <stdio.h>
#include <stdlib.h>// 抽象产品Department
typedef struct Department
{void (*Insert)(void *);struct Department *(*GetDepartment)(void *);
} Department;// 抽象产品User
typedef struct User
{void (*Insert)(void *);struct User *(*GetUser)(void *);
} User;// 抽象工厂
typedef struct AbstractFactory
{User *(*CreateUser)(void);Department *(*CreateDepartment)(void);
} AbstractFactory;// 具体产品SqlserDepartment
typedef struct SqlserDepartment
{Department base;
} SqlserDepartment;void SqlserDepartment_Insert(void *dept)
{printf("在SQL Server 中给 Department 表增加一条记录!\n");
}Department *SqlserDepartment_GetDepartment(void *dept)
{printf("在SQL Server 中根据ID 得到 Department 表一条记录!\n");return NULL;
}// 具体产品AccessDepartment
typedef struct AccessDepartment
{Department base;
} AccessDepartment;void AccessDepartment_Insert(void *dept)
{printf("在 Access 中给 Department 表增加一条记录!\n");
}Department *AccessDepartment_GetDepartment(void *dept)
{printf("在 Access 中根据ID 得到 Department 表一条记录!\n");return NULL;
}// 具体产品SqlserverUser
typedef struct SqlserverUser
{User base;
} SqlserverUser;void SqlserverUser_Insert(void *user)
{printf("在SQL Server 中给 User表增加一条记录!\n");
}User *SqlserverUser_GetUser(void *user)
{printf("在SQL Server 中根据ID 得到 User表一条记录!\n");return NULL;
}// 具体产品AccessUser
typedef struct AccessUser
{User base;
} AccessUser;void AccessUser_Insert(void *user)
{printf("在 Access 中给 User表增加一条记录!\n");
}User *AccessUser_GetUser(void *user)
{printf("在 Access 中根据ID 得到 User表一条记录!\n");return NULL;
}// 具体工厂SqlServerFactory
typedef struct SqlServerFactory
{AbstractFactory base;
} SqlServerFactory;User *SqlServerFactory_CreateUser(void *unused)
{SqlserverUser *user = (SqlserverUser *)malloc(sizeof(SqlserverUser));user->base.Insert = SqlserverUser_Insert;user->base.GetUser = SqlserverUser_GetUser;return (User *)user;
}Department *SqlServerFactory_CreateDepartment(void *unused)
{SqlserDepartment *dept = (SqlserDepartment *)malloc(sizeof(SqlserDepartment));dept->base.Insert = SqlserDepartment_Insert;dept->base.GetDepartment = SqlserDepartment_GetDepartment;return (Department *)dept;
}// 具体工厂AccessFactory
typedef struct AccessFactory
{AbstractFactory base;
} AccessFactory;User *AccessFactory_CreateUser()
{AccessUser *user = (AccessUser *)malloc(sizeof(AccessUser));user->base.Insert = AccessUser_Insert;user->base.GetUser = AccessUser_GetUser;return (User *)user;
}Department *AccessFactory_CreateDepartment()
{AccessDepartment *dept = (AccessDepartment *)malloc(sizeof(AccessDepartment));dept->base.Insert = AccessDepartment_Insert;dept->base.GetDepartment = AccessDepartment_GetDepartment;return (Department *)dept;
}int main()
{AbstractFactory *factory = (AbstractFactory *)malloc(sizeof(AccessFactory));((AccessFactory *)factory)->base.CreateUser = AccessFactory_CreateUser;((AccessFactory *)factory)->base.CreateDepartment = AccessFactory_CreateDepartment;User *iu = factory->CreateUser();iu->Insert(iu);iu->GetUser(iu);Department *id = factory->CreateDepartment();id->Insert(id);id->GetDepartment(id);free(factory);free(iu);free(id);return 0;
}
总结
在工厂方法模式的基础上,在对应的工厂类中新增了相关产品的创建函数。(其实并不是很方便!!)
扩展:
简单工厂模式似乎更简便点,但是switch部分实在是破坏了开闭原则,可以使用“反射”(即动态识别类型)
C++可以使用静态函数指针数组来实现
参考:https://blog.csdn.net/klarclm/article/details/7352101
相关文章:
抽象工厂模式(大话设计模式)C/C++版本
抽象工厂模式 C 参考:https://www.cnblogs.com/Galesaur-wcy/p/15927110.html #include <iostream> using namespace std;// 抽象产品Department ,定义具体产品的公共接口 class Department { public:virtual ~Department() default;virtual void Insert()…...
springboot宠物医院信息管理系统-计算机毕业设计源码04164
摘 要 现如今在中国,随着人民生活质量的逐渐提高,以及人民群众消费能力的日渐增长,各种各样的家养小动物,已经逐渐成为人类越来越亲密的生活伴侣。并且,现如今社会竞争及其激烈,人们的生活节奏越发急促、紧…...
Leetcode Hot100之哈希表
1. 两数之和 题目描述 给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并返回它们的数组下标。 你可以假设每种输入只会对应一个答案。但是,数组中同一个元素在答案里不能重复出现思路…...
Vision Transformer with Sparse Scan Prior
摘要 https://arxiv.org/pdf/2405.13335v1 In recent years, Transformers have achieved remarkable progress in computer vision tasks. However, their global modeling often comes with substantial computational overhead, in stark contrast to the human eye’s eff…...
笔记-python 中BeautifulSoup入门
在前面的例子用,我用了BeautifulSoup来从58同城抓取了手机维修的店铺信息,这个库使用起来的确是很方便的。本文是BeautifulSoup 的一个详细的介绍,算是入门把。文档地址:http://www.crummy.com/software/BeautifulSoup/bs4/doc/ …...
Tomcat Websocket应用实例研究
概述 本文介绍了如何根据Tomcat给出的websocket实例,通过对实例的学习,定制自己基于websocket的应用。 环境及版本: Ubuntu 22.04.4 LTSApache Tomcat/10.1.20openjdk 11.0.23 2024-04-16浏览器:Chrome 相关资源及链接 Class…...
leetcode-11-二叉树前中后序遍历以及层次遍历
一、递归版 前序遍历 (先根遍历) 中左右 class Solution {public List<Integer> preorderTraversal(TreeNode root) {List<Integer> result new ArrayList<Integer>();preorder(root, result);return result;}public void preorder…...
Python基础学习笔记(十一)——集合
目录 一、集合的介绍与创建二、集合的存储原理三、元素的修改1. 添加元素2. 删除元素 四、集合的运算五、集合的判定 一、集合的介绍与创建 集合(set),一种可变、无序、不重复的数据结构,由大括号{}内、用逗号分隔的一组元素组成。…...
FineReport
1.FineReport 官网 :FineReport产品简介- FineReport帮助文档 - 全面的报表使用教程和学习资料 下载地址 免费下载FineReport - FineReport报表官网 FineReport是一款用于报表制作,分析和展示的工具。 普通模板:是 FineReport 最常用…...
嵌入式就业前景好么
嵌入式就业前景在当前环境下是较为乐观的,以下是对嵌入式就业前景的详细分析: 广泛应用领域:嵌入式系统广泛应用于智能家居、医疗设备、航空航天等领域。随着物联网(IoT)的快速发展,预计到2024年ÿ…...
为啥找对象千万别找大厂男,还好我不是大厂的。。
网上看到一大厂女员工发文说:找对象千万别找大厂男,理由说了一大堆,无非就是大厂男为了逃避带娃,以加班为由宁愿在工位上玩游戏也不愿回家。当然这种观点有的人赞同有的人反对。 网友精彩评论: --------------下面是今…...
如何查看k8s中service的负载均衡策略
在Kubernetes中,Service的负载均衡策略一般由kube-proxy负责,kube-proxy使用iptables或IPVS规则进行负载均衡。默认情况下,kube-proxy使用的是轮询(Round Robin)策略,但是在使用IPVS模式时,可以…...
Linux-DNS域名解析服务01
BIND 域名服务基础 1、DNS(Domain Name System)系统的作用及类型 整个 Internet 大家庭中连接了数以亿计的服务器、个人主机,其中大部分的网站、邮件等服务器都使用了域名形式的地址,如 www.google.com、mail.163.com 等。很显然…...
[c++刷题]贪心算法.N01
题目如上: 首先通过经验分析,要用最少的减半次数,使得数组总和减少至一半以上,那么第一反应就是每次都挑数组中最大的数据去减半,这样可以是每次数组总和值减少程度最大化。 代码思路:利用大根堆去找数据中的最大值,…...
推荐常用的三款源代码防泄密软件
三款源代码防泄密软件——安秉源代码加密、Virbox Protector 和 MapoLicensor——确实各自在源代码保护的不同方面有其专长。这些软件可以满足企业对于源代码保护的三大需求:防止泄露、防止反编译和防止破解。 安秉源代码加密: 专注于源代码文件的加密&…...
Android 13 高通设备热点低功耗模式(2)
前言 之前写过一篇文章:高通热点被IOS设备识别为低数据模式,该功能仿照小米的低数据模式写的,散发的热点可以达到被IOS和小米设备识别为低数据模式。但是发现IOS设备如果后台无任何网络请求的时候,息屏的状态下过一会,会自动断开热点的连接。 分析 抓取设备的热点相关的…...
web前端任职条件:全面解析
web前端任职条件:全面解析 在当今数字化快速发展的时代,Web前端技术已经成为互联网行业不可或缺的一部分。作为一名Web前端开发者,需要具备哪些任职条件呢?本文将从四个方面、五个方面、六个方面和七个方面为您深入剖析。 四个方…...
分析医药零售数据该用哪个BI数据可视化工具?
数据是企业决策的重要依据,可以用于现代企业大数据可视化分析的BI工具有很多,各有各擅长的领域。那么哪个BI数据可视化工具分析医药零售数据又好又快? 做医药零售数据分析首推奥威BI数据可视化工具! 奥威BI数据可视化工具做医药…...
如何使用芯片手册做软件开发?
在阅读和利用芯片手册进行软件开发时,你应该关注以下几个关键点: 引脚功能:了解芯片上每个引脚的功能,包括它们可以被配置为输入还是输出,以及它们支持的特殊功能,如模拟输入、PWM输出、中断等。 寄存器映…...
基于深度学习的文本翻译
基于深度学习的文本翻译 基于深度学习的文本翻译,通常称为神经机器翻译(Neural Machine Translation, NMT),是近年来在自然语言处理(NLP)领域取得显著进展的技术。NMT通过使用深度神经网络来自动学习和翻译…...
Flask RESTful 示例
目录 1. 环境准备2. 安装依赖3. 修改main.py4. 运行应用5. API使用示例获取所有任务获取单个任务创建新任务更新任务删除任务 中文乱码问题: 下面创建一个简单的Flask RESTful API示例。首先,我们需要创建环境,安装必要的依赖,然后…...
树莓派超全系列教程文档--(61)树莓派摄像头高级使用方法
树莓派摄像头高级使用方法 配置通过调谐文件来调整相机行为 使用多个摄像头安装 libcam 和 rpicam-apps依赖关系开发包 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 配置 大多数用例自动工作,无需更改相机配置。但是,一…...
学校招生小程序源码介绍
基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码,专为学校招生场景量身打造,功能实用且操作便捷。 从技术架构来看,ThinkPHP提供稳定可靠的后台服务,FastAdmin加速开发流程,UniApp则保障小程序在多端有良好的兼…...
Module Federation 和 Native Federation 的比较
前言 Module Federation 是 Webpack 5 引入的微前端架构方案,允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...
uniapp 集成腾讯云 IM 富媒体消息(地理位置/文件)
UniApp 集成腾讯云 IM 富媒体消息全攻略(地理位置/文件) 一、功能实现原理 腾讯云 IM 通过 消息扩展机制 支持富媒体类型,核心实现方式: 标准消息类型:直接使用 SDK 内置类型(文件、图片等)自…...
SQL Server 触发器调用存储过程实现发送 HTTP 请求
文章目录 需求分析解决第 1 步:前置条件,启用 OLE 自动化方式 1:使用 SQL 实现启用 OLE 自动化方式 2:Sql Server 2005启动OLE自动化方式 3:Sql Server 2008启动OLE自动化第 2 步:创建存储过程第 3 步:创建触发器扩展 - 如何调试?第 1 步:登录 SQL Server 2008第 2 步…...
图解JavaScript原型:原型链及其分析 | JavaScript图解
忽略该图的细节(如内存地址值没有用二进制) 以下是对该图进一步的理解和总结 1. JS 对象概念的辨析 对象是什么:保存在堆中一块区域,同时在栈中有一块区域保存其在堆中的地址(也就是我们通常说的该变量指向谁&…...
ThreadLocal 源码
ThreadLocal 源码 此类提供线程局部变量。这些变量不同于它们的普通对应物,因为每个访问一个线程局部变量的线程(通过其 get 或 set 方法)都有自己独立初始化的变量副本。ThreadLocal 实例通常是类中的私有静态字段,这些类希望将…...
手动给中文分词和 直接用神经网络RNN做有什么区别
手动分词和基于神经网络(如 RNN)的自动分词在原理、实现方式和效果上有显著差异,以下是核心对比: 1. 实现原理对比 对比维度手动分词(规则 / 词典驱动)神经网络 RNN 分词(数据驱动)…...
Web APIS Day01
1.声明变量const优先 那为什么一开始前面就不能用const呢,接下来看几个例子: 下面这张为什么可以用const呢?因为复杂数据的引用地址没变,数组还是数组,只是添加了个元素,本质没变,所以可以用con…...
