抽象工厂模式(大话设计模式)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

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

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

相关文章

CC2500和CC1101移植说明

主要通过如何移植、移植注意、关于芯片配置、如何生成导出配置四大步骤来说明CC2500和CC1101移植 首先通过下图1这个宏进行选择 如何移植 要移植的部分在 CC2500_hal.c 和 CC2500_hal.h中, 搜索 "//移植" 就可以定位到 库 所需的依赖, 需要根据 您的环境实现这些…

(done) 什么是 perplexity 困惑度?

参考&#xff1a;https://www.youtube.com/watch?vB_2bntDYano 困惑度 perplexity 是一种用来衡量语言模型性能的度量&#xff0c;类似于交叉熵。 困惑度越低越好&#xff0c;越低说明一个模型越好。 一个典型的公式在下面&#xff1a;

Python学习打卡:day06

day6 笔记来源于&#xff1a;黑马程序员python教程&#xff0c;8天python从入门到精通&#xff0c;学python看这套就够了 目录 day648、函数综合案例49、数据容器入门50、列表的定义语法51、列表的下标索引1、列表的下标&#xff08;索引&#xff09;2、列表的下标&#xff08…

CCAA质量管理【学习笔记】​​ 备考知识点笔记(四)

第四节 质量非数据资料分析的基础工具 1 关 联 图 2.1 概念 所谓关联图&#xff0c;就是对关系复杂而相互纠缠的问题&#xff0c;依据原因—结果或目的一手段等关系&#xff0c; 在逻辑上用箭头把各要素之间的因果关系连接起来&#xff0c;厘清复杂问题、整理语言文字资料…

技术转管理,是灾难还是奇迹?

深耕技术or转战管理&#xff1f;this is a question! 如果你还没有想好&#xff0c;那请继续往下看&#xff01; 技术专家&#xff1a;技术前瞻者、方案构建者、难题破解者、团队聚核者 管理专家&#xff1a;战略规划者、高效组织者、变革引领者、团队建设者 特点和重心都不在…

iPad键鼠充电otg转接器 | LDR6020解决方案

随着科技的快速发展&#xff0c;iPad已经成为我们日常生活中不可或缺的一部分。它不仅是一个娱乐工具&#xff0c;更是一个高效的生产力工具。为了更好地满足用户的需求&#xff0c;iPad支持在充电的同时连接鼠标和键盘&#xff0c;极大地提升了使用的便捷性和效率。 iPad键鼠同…

干部选拔任用的六条原则

在干部选拔任用的过程中&#xff0c;为确保选拔出的干部能够真正符合党和人民的期望&#xff0c;必须遵循以下六条原则&#xff1a; 一、党管干部原则 党管干部原则是指在整个干部选拔任用过程中&#xff0c;党要发挥总揽全局、协调各方的领导作用&#xff0c;确保选拔出的干…

省去烦恼!轻松实现一台电脑登录多个微信号的秘诀揭秘!

你知道如何在同一台电脑上登录多个微信号&#xff0c;并实现聚合聊天吗&#xff1f; 今天&#xff0c;我将分享一个多微管理神器——个微管理系统&#xff0c;帮助你解决这一问题&#xff01; 1、多号同时登录&#xff0c;聚合聊天 无论你有多少个微信号&#xff0c;都可以一…

2024年【安全生产监管人员】试题及解析及安全生产监管人员考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 安全生产监管人员试题及解析是安全生产模拟考试一点通总题库中生成的一套安全生产监管人员考试试题&#xff0c;安全生产模拟考试一点通上安全生产监管人员作业手机同步练习。2024年【安全生产监管人员】试题及解析及…

智慧停车场定位导航系统的实现与优化

随着城市车辆数量的激增&#xff0c;大型停车场的运营挑战日益严峻。维小帮停车场导航定位系统通过集成先进的室内定位技术和大数据分析&#xff0c;为停车场提供了一套高效、便捷的导航定位服务方案&#xff0c;有效解决了停车难、找车难、管理难等问题。 一、系统核心功能的…

Zookeeper: 配置参数解读

Zookeeper中的配置文件zoo.cfg中参数含义解读如下&#xff1a; tickTime&#xff1a;通信心跳时间&#xff0c;Zookeeper服务器与客户端心跳时间&#xff0c;单位毫秒。 initLimit: LF初始通信时限 Leader和Follower初始连接时能容忍的最多心跳数。 syncLimit: LF同步通信时…

dos下命令行批量修改文件名

1、文件夹下执行获取所有文件名: dir *.* /b > AAA.txt 2、复制aaa.txt文件内容&#xff0c;在excel 下加工成这种格式&#xff08;多行脚本&#xff09;&#xff1a; rename IMG_1561.JPG 20240614会考_005.jpg 3、执行多行脚本

STM32学习和实践笔记(35):内部温度传感器实验

1.STM32F1内部温度传感器介绍 1.1 STM32F1内部温度传感器简介 STM32F1内部含有一个温度传感器&#xff0c;可用来测量 &#xff08;STM32芯片的&#xff09;CPU 及周围的温度(TA)。&#xff08;实际并不用来测周围的温度&#xff0c;仅用来测试CPU的温度&#xff09; 此温度传…

以keepalived为例说明程序不能正常被gdb调试的原因

现象 通过gdb att $keepalived_pid发起对当前运行keepalived的调试&#xff1b; 在放行keepalived继续执行后&#xff0c;想通过CtrlC按键中断执行&#xff0c;观察下被调试程序的当前内部状态&#xff0c; 但是&#xff0c;在终端输入CtrlC后&#xff0c;导致keepalived被调…

通过语言大模型来学习LLM和LMM(四)

一、大模型学习 新的东西&#xff0c;学习的东西就是多&#xff0c;而且最简单最基础的都需要学习&#xff0c;仿佛一点基础知识都要细嚼慢咽&#xff0c;刨根问底&#xff0c;再加上一顿云里雾里的吹嘘&#xff0c;迷迷糊糊的感觉高大上。其实就是那么一回事。再过一段时日&a…

【Mongodb-02】springboot整合mongodb(详解)

springBoot整和mongodb 一&#xff0c;springboot整合mongodb1&#xff0c;依赖加入2&#xff0c;yml文件配置3&#xff0c;_class 字段过滤(可选)4&#xff0c;实体类定义5&#xff0c;索引创建6&#xff0c;数据插入6.1&#xff0c;insert方式6.2&#xff0c;使用save的方式实…

awtk如何实现键盘和输入框

1.创建默认键盘 新建窗体-keyboard 2.新建编辑框 3.设置编辑框属性 4.点击编辑框即可打开默认键盘&#xff0c;若想修改键盘样式可以在默认键盘修改或自定义键盘 5.获取输入字符 widget_t* wifi_edit widget_lookup(win, "edit", TRUE);//获取单行编辑控件 widge…

解决Windows中端口占用导致服务启动失败

解决Windows中端口占用导致服务启动失败 在cmd窗口中使用netstat -ano | findstr "3306"来查看哪个线程占用了3306端口。 下面的图片里面表示一个pid为5196的进程占用了端口 接着可以在cmd窗口中使用tasklist | findstr "5196" 根据pid查询进程名称 通过…

0-1 构建用户画像数仓

目录 前言 一、用户画像概述 1.1 用户画像 1.2 用户标签 1.3 用户群组 二、建设标签和标签体系 2.1 标签体系 2.1.1 统计类标签 2.1.2 规则类标签 2.1.3 机器学习挖掘类标签 2.2 标签建设流程 2.2.1 需求收集与分析 2.2.2 产出标签需求文档 2.2.3 标签的开发 H…

ffmpeg封装和解封装介绍-(8)解封装和封装重构

头文件&#xff1a; xformat.h #pragma once/// 封装和解封装基类#include <mutex> struct AVFormatContext; struct AVCodecParameters; struct AVPacket; struct XRational {int num; ///< Numeratorint den; ///< Denominator }; class XFormat { public:/// &…