C++设计模式|创建型 2.工厂模式

1.简单工厂思想

简单工厂模式不属于23种设计模式之⼀,更多的是⼀种编程习惯。它的核心思想是将产品的创建过程封装在⼀个⼯⼚类中,把创建对象的流程集中在这个⼯⼚类⾥⾯。卡码网将其结构描述为下图所示的情况:

简单⼯⼚模式包括三个主要⻆⾊,⼯⼚类、抽象产品类、具体产品类。它们的职责如下:

  • ⼯⼚类:负责创建产品,根据传递的不同参数创建不同的产品示例。
  • 抽象产品类:由工厂类提供的接口,⽐如上图中的 Shape 接⼝,描述产品的通⽤⾏为。
  • 具体产品类:实现抽象产品接⼝或继承抽象产品类,⽐如上⾯的 Circle 类和 Square 类,具体产品通过简单⼯⼚类的 if-else 逻辑来实例化。

 简单工厂的优点是简化了客户端的操作,客户端可以调用工厂方法来获取具体产品,而无需直接与具体产品类交互,降低了耦合,但是有一个很大的问题就是不够灵活,如果需要添加新的产品,就需要修改工厂类的代码

2.什么是工厂模式?

简单工厂的思想中只有一个工厂类,用于创建所有的产品,如果需要添加新的产品,就需要修改工厂类的代码。而工厂模式引入了抽象工厂和具体工厂的概念,每个具体工厂只负责创建一个具体产品,添加新产品时只需要添加新的工厂类,支持扩展,复合开闭原则。

工厂方法模式分为以下几个角色:

  • 抽象工厂:一个接口,包含一个抽象的工厂方法,该方法用于创建产品对象
  • 具体工厂:实现抽象工厂接口,创建具体的产品。
  • 抽象产品类:定义产品的接口。
  • 具体产品类:实现抽象产品接口,是工厂创建的对象。

 工厂模式的示意图如下图所示,实际的生产系统所管理的是对应的工厂,工厂负责产品的生产。当有具体需求传入时,生产提供根据需求创建对应的工厂,这些工厂去完成需求产品的生产。在有新的产品时,只需要扩展一下产品的内容及其对应的工厂即可,原来的代码无需更改,只需要往里面添加新的模块即可,非常灵活。

 3.C++工厂模式

【设计模式专题之工厂方法模式】2.积木工厂 (kamacoder.com)icon-default.png?t=N7T8https://kamacoder.com/problempage.php?pid=1076

题目描述:

小明家有两个工厂,一个用于生产圆形积木,一个用于生产方形积木,请你帮他设计一个积木工厂系统,记录积木生产的信息。

输入描述

输入的第一行是一个整数 N(1 ≤ N ≤ 100),表示生产的次数。 

接下来的 N 行,每行输入一个字符串和一个整数,字符串表示积木的类型。积木类型分为 "Circle" 和 "Square" 两种。整数表示该积木生产的数量

输出描述

对于每个积木,输出一行字符串表示该积木的信息。

输入示例

3
Circle 1
Square 2
Circle 1

 输出示例

Circle Block
Square Block
Square Block
Circle Block

代码实现: 

先实现抽象产品类和具体产品类:

//抽象积木产品类,定义生产具体积木的接口produce()
class Block{
public:
    //写成纯虚函数,一方面将该类变成抽象类,另一方面方便具体产品类实现多态
    virtual void produce() = 0;
};


//具体圆形积木产品类
class CircleBlock: public Block{
public:
    //重写父类虚函数,使用override关键字编译器会检查重写是否正确,不是非要加
    void produce() override {
        cout<<"Circle Block"<<endl;
    }
};

//具体方形积木产品类
class SquareBlock: public Block{
public:
    //重写父类虚函数
    void produce() override
    {
        cout<<"Square Block"<<endl;
    }
};

再实现抽象工厂类和具体工厂类,其中具体工厂类要调用具体产品的生产函数,所以工厂类写在了产品类的后面:

//抽象工厂类,提供一个抽象的接口用于创建对象。具体哪种对象还得用多态实现
class BlockFactory{
public:
    //抽象的接口。 使用积木的基类指针作为返回值,其可以接收圆形或方形的积木对象(C++多态允许)
    virtual Block* createBlock() = 0;
};

//具体圆形积木工厂类
class CircleBlockFactory:public BlockFactory{
public:  
    //重写接口函数
    Block* createBlock() override
    {
        return new CircleBlock();  //堆区开辟内存存放积木对象
    }
};

//具体方形积木工厂类
class SquareBlockFactory:public BlockFactory{
public:  
    //重写接口函数
    Block* createBlock() override
    {
        return new SquareBlock();  //堆区开辟内存存放积木对象
    }
};

建立积木工厂系统。该系统根据输入要求去建立相对应的工厂,让该工厂去生产具体产品:

//建立积木工厂系统
class BlockManageSystem{
private:
    //使用Block*动态数组来记录积木信息
    vector<Block *> blocks;
public:
    //根据输入,建立积木生产函数。根据具体产品类型调用相对应的产品工厂
    void produceBlocks(BlockFactory* factory, int num)
    {
        for(int i=0; i<num; i++)
        {
            Block* block = factory->createBlock();
            this->blocks.push_back(block);
            block->produce();
        }
    }
    
    //由于具体积木存放在堆区,所以要使用delete进行内存释放
    ~BlockManageSystem(){
        for(Block* block : blocks)
        {
            delete block;
        }
    }
    
    //获取所有积木。第一个const修饰返回的引用,即使得到积木,也不能对积木进行修改。
    //第二个const修饰this指针,保证this指针不能对积木进行修改
    const vector<Block*> & getBlocks()const{
        return blocks;
    }
    
};

总体代码:

#include<iostream>
#include<vector>
#include<string>
using namespace std;

//按照工厂模式的组成,依次实现抽象产品类、具体产品类、抽象工厂类、具体工厂类

//抽象积木产品类,定义生产具体积木的接口produce()
class Block{
public:
    //写成纯虚函数,一方面将该类变成抽象类,另一方面方便具体产品类实现多态
    virtual void produce() = 0;
};


//具体圆形积木产品类
class CircleBlock: public Block{
public:
    //重写父类虚函数
    void produce() override {
        cout<<"Circle Block"<<endl;
    }
};

//具体方形积木产品类
class SquareBlock: public Block{
public:
    //重写父类虚函数,使用override关键字编译器会检查重写是否正确,不是非要加
    void produce() override
    {
        cout<<"Square Block"<<endl;
    }
};


//抽象工厂类,提供一个抽象的接口用于创建对象。具体哪种对象还得用多态实现
class BlockFactory{
public:
    //抽象的接口。 使用积木的基类指针作为返回值,其可以接收圆形或方形的积木对象(C++多态允许)
    virtual Block* createBlock() = 0;
};

//具体圆形积木工厂类
class CircleBlockFactory:public BlockFactory{
public:  
    //重写接口函数
    Block* createBlock() override
    {
        return new CircleBlock();  //堆区开辟内存存放积木对象
    }
};

//具体方形积木工厂类
class SquareBlockFactory:public BlockFactory{
public:  
    //重写接口函数
    Block* createBlock() override
    {
        return new SquareBlock();  //堆区开辟内存存放积木对象
    }
};


//建立积木工厂系统
class BlockManageSystem{
private:
    //使用Block*动态数组来记录积木信息
    vector<Block *> blocks;
public:
    //根据输入,建立积木生产函数。根据具体产品类型调用相对应的产品工厂
    void produceBlocks(BlockFactory* factory, int num)
    {
        for(int i=0; i<num; i++)
        {
            Block* block = factory->createBlock();
            this->blocks.push_back(block);
            block->produce();
        }
    }
    
    //由于具体积木存放在堆区,所以要使用delete进行内存释放
    ~BlockManageSystem(){
        for(Block* block : blocks)
        {
            delete block;
        }
    }
    
    //获取所有积木。第一个const修饰返回的引用,即使得到积木,也不能对积木进行修改。
    //第二个const修饰this指针,保证this指针不能对积木进行修改
    const vector<Block*> & getBlocks()const{
        return blocks;
    }
    
};

int main()
{
    int produceNum;  //生产次数
    cin>>produceNum;
    
    BlockManageSystem mySystem;  //创建积木工厂系统
    
    
    for(int i=0; i< produceNum; i++)
    {
        string blockType;
        int blockNum;
        //读取生产积木的类型和数量
        cin>>blockType>>blockNum;
        if(blockType == "Circle") 
        {
            //需要使用工厂系统调用圆形积木工厂启动生产
            mySystem.produceBlocks(new CircleBlockFactory(),blockNum);
        }
        else if (blockType == "Square") 
        {
            //需要使用工厂系统调用方形积木工厂启动生产
            mySystem.produceBlocks(new SquareBlockFactory(),blockNum);
        }
    }
    return 0;
}

4.工厂模式应用场景

工厂方法模式使得每个工厂类的职责单一每个工厂只负责创建一种产品,当创建对象涉及一系列复杂的初始化逻辑,而这些逻辑在不同的子类中可能有所不同时,可以使用工厂方法模式将这些初始化逻辑封装在子类的工厂中。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

zabbix 自动发现与自动注册 部署 zabbix 代理服务器

zabbix 自动发现&#xff08;对于 agent2 是被动模式&#xff09; zabbix server 主动的去发现所有的客户端&#xff0c;然后将客户端的信息登记在服务端上。 缺点是如果定义的网段中的主机数量多&#xff0c;zabbix server 登记耗时较久&#xff0c;且压力会较大。1.确保客户端…

uboot的移植

文章目录 一、官方uboot移植1.Uboot系统复制到Ubuntu系统2.解压Uboot系统3.编译Uboot系统4.生成可执行文件5.将u-boot.bin烧录到SD卡6.SD卡插入到板子&#xff0c;启动方式选择SD卡7.复位板子&#xff0c;查看打印信息&#xff0c;编译时间是否正常 二、根据官方提供的uboot添加…

frp 内网穿透配置(v0.55.1 版本)

注意&#xff1a;从 [v0.52.0] 版本开始&#xff0c;配置文件由 frps.ini 改成了 frps.toml 一种快速反向代理&#xff0c;可帮助您将 NAT 或防火墙后面的本地服务器暴露给 Internet。 GitHub 地址 &#xff1a; github.com/fatedier/fr… 下载之后如果碰到杀毒软件报毒&#x…

富文本在线编辑器 - tinymce

tinymce 项目是一个比较好的富文本编辑器. 这里有个小demo, 下载下来尝试一下, 需要配置个本地服务器才能够访问, 我这里使用的nginx, 下面是我的整个操作过程: git clone gitgitee.com:chick1993/layui-tinymce.git cd layui-tinymcewget http://nginx.org/download/nginx-1.…

00_Qt概述以及如何创建一个QT新项目

Qt概述 1.Qt概述1.1 什么是Qt1.2 Qt的发展史1.3 支持的平台1.4 Qt版本1.5 Qt的下载与安装1.6 Qt的优点 2.QT新项目创建3.pro文件4.主函数5.代码命名规范和快捷键 1.Qt概述 1.1 什么是Qt Qt是一个跨平台的C图形用户界面应用程序框架。它为应用程序开发者提供建立艺术级图形界面…

【一竞技CS2】VP战队官宣签下electroNic取代mir

1、近日VP战队官宣签下electroNic&#xff0c;以取代阵容中的mir。 electroNic自己也表示&#xff1a;“VP是一支顶级队伍。阵容核心曾赢得Major冠军&#xff0c;所有队员都处于巅峰状态并且时刻准备着去争夺冠军。我们有着一样的雄心壮志。 此外我还对和Jame很感兴趣&#xf…

解决nginx日志过大问题

1. 问题点 nginx默认的日志在logs/access.log&#xff0c;并且是一直累加写入&#xff0c;时间长了就会非常大&#xff0c;占用过多的硬盘&#xff0c;如果强行删除是很不友好的&#xff0c;需要重启服务&#xff1b; 2. 文件分割 上图文件已经达到了十个G左右 处理的思路肯定…

AI大模型探索之路-应用篇14:认识国产开源大模型GLM

目录 前言 一、国产主流大模型概览 1. 国内主流大模型清单 2. 主流大模型综合指数 3. 大语言模型评测榜单 二、GLM大模型介绍 三、GLM大模型发展历程 四、GLM家族之基座模型GLM-130B 五、GLM家族之ChatGLM3 六、GLM家族之WebGLM 七、GLM家族之CogVLM 1. CogVLM 2. …

2024五一杯数学建模A题思路分析

文章目录 1 赛题思路2 比赛日期和时间3 组织机构4 建模常见问题类型4.1 分类问题4.2 优化问题4.3 预测问题4.4 评价问题 5 建模资料 1 赛题思路 (赛题出来以后第一时间在CSDN分享) https://blog.csdn.net/dc_sinor?typeblog 2 比赛日期和时间 报名截止时间&#xff1a;2024…

P9241 [蓝桥杯 2023 省 B] 飞机降落

原题链接&#xff1a;[蓝桥杯 2023 省 B] 飞机降落 - 洛谷 目录 1. 题目描述 2. 思路分析 3. 代码实现 1. 题目描述 2. 思路分析 dfs全排列的变形题。 因为最后问飞机是否降落&#xff0c;并且一架飞机降落完毕时另一架飞机才能降落。所以我们设置dfs的两个变量cnt为安全…

解决EasyPoi导入Excel获取不到第一列的问题

文章目录 1. 复现错误2. 分析错误2.1 导入的代码2.2 DictExcel实体类2.2 表头和标题 3. 解决问题 1. 复现错误 使用EasyPoi导入数据时&#xff0c;Excel表格如下图&#xff1a; 但在导入时&#xff0c;出现如下错误&#xff1a; name为英文名称&#xff0c;在第一列&#xff0c…

Java代码基础算法练习-水仙花数-2024.04.17

任务描述&#xff1a; 水仙花数也被称为超完全数字不变数、自恋数、自幂数、阿姆斯壮数或阿姆斯特朗数。水仙花数是 指一个 3 位数&#xff0c;它的每个位上的数字的3次幂之和等于它本身。 例如: 1的3次方 5的3次方 …

计算机网络的七层模型

序 OSl(Open System Interconnect)&#xff0c;即开放式系统互联。一般都叫OSI参考模型。在网络编程中最重要的模型就是OSI七层网络模型和TCP/IP四层网络模型 一、OSI七层参考模型以及功能概述 二、各层的具体职能以及实际应用 1.应用层&#xff1a; OSI参考模型中最接近用…

最新的网易星球GEC挖矿系统修复版 章鱼星球挖矿系统源码 区块链虚拟币交易源码 基于ThinkPHP5开发

区块链系统介绍 2018.12.10更新增加聚合数据短信接口 2018.11.19更新增加短信宝接口 2018.08.17修复Linux系统搭建验证码不显示问题 2018.08.09修复后台某处溢出数据库账号密码BUG 2018.08.06修复票卷BUG 源码介绍&#xff1a; 区块链系统中用户共九个等级&#xff0c;依…

【Git】生成patch和应用patch

生成patch 将本地所有修改打成补丁 git diff > /tmp/xxx.patch将本地对某个文件的修改打成补丁 git diff test/1.txt > /tmp/1.patch将某一次提交的修改内容打成补丁 -1表示只为单个提交创建patch&#xff0c;-o表示输出patch的文件夹路径&#xff0c;默认是用提交的…

轻松查询车辆信息的全能接口

在当今社会&#xff0c;车辆已经成为人们出行的重要工具之一。当我们在二手车买卖、事故处理或者其他需要查询车辆详细信息的情况下&#xff0c;我们通常需要耗费大量时间和精力去收集相关的资料。幸好&#xff0c;有了车辆信息查询接口&#xff0c;我们可以通过输入车架号vin来…

20240416,对象初始化和清理,对象模型和THIS指针

哈哈哈乌龟越狱了 目录 2.5 深拷贝&浅拷贝 2.6 初始化列表 2.7 类对象作为类成员 2.8 静态成员 2.9 成员变量和成员函数分开存储 2.10 THIS指针的用途 2.11 空指针访问成员函数 2.12 COSNT修饰成员函数 2.5 深拷贝&浅拷贝 浅拷贝&#xff1a;简单的赋值拷贝…

leetcode-合并两个有序链表

目录 题目 图解 方法一 方法二 代码(解析在注释中) 方法一 ​编辑方法二 题目 将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 示例 1&#xff1a; 输入&#xff1a;l1 [1,2,4], l2 [1,3,4] 输出&#xff1a;[1,1…

第11章 数据仓库和数据智能知识点梳理

第11章 数据仓库和数据智能知识点梳理&#xff08;附带页码&#xff09; ◼ 数据仓库&#xff08;Data Warehouse&#xff0c;DW&#xff09;&#xff1a;始于 20 世纪 80 年代&#xff0c;发展于 20 世纪 90 年代&#xff0c;后与商务智能&#xff08;Business Inteligence,BI…

MAC上如何将某个目录制作成iso格式磁盘文件,iso文件本质是什么?以及挂载到ParallelDesktop中?(hdiutil makehybrid )

背景 ParallelsDesktop没有安装ParallelsTools的无法共享目录&#xff0c;可以通过ParallelsDesktop提供CD磁盘的方式共享进去 命令 # 准备文档 mkdir mytestdir cp xxx mytestdir# 生成iso hdiutil makehybrid -o output.iso mytestdir -iso -joliethdiutil是MAC提供的磁盘…