C/C++实现工厂模板设计模式(Factory Pattern)

工厂模式是一种创建型设计模式,它提供了一种创建对象的方式,而无需指定具体的类。工厂模式通过定义一个创建对象的接口,让子类决定实例化哪一个类。工厂模式可以分为简单工厂模式、工厂方法模式和抽象工厂模式。

下面是一个使用C++实现的简单工厂模式的示例:

#include <iostream>
#include <memory>// 产品基类
class Product {
public:virtual void use() = 0;virtual ~Product() = default;
};// 具体产品A
class ProductA : public Product {
public:void use() override {std::cout << "Using Product A" << std::endl;}
};// 具体产品B
class ProductB : public Product {
public:void use() override {std::cout << "Using Product B" << std::endl;}
};// 工厂类
class Factory {
public:// 创建产品的方法static std::unique_ptr<Product> createProduct(const std::string& type) {if (type == "A") {return std::make_unique<ProductA>();} else if (type == "B") {return std::make_unique<ProductB>();} else {return nullptr;}}
};int main() {// 使用工厂创建产品Astd::unique_ptr<Product> productA = Factory::createProduct("A");if (productA) {productA->use();}// 使用工厂创建产品Bstd::unique_ptr<Product> productB = Factory::createProduct("B");if (productB) {productB->use();}return 0;
}

代码说明:

  1. Product 是产品的基类,定义了产品的接口 use()
  2. ProductAProductB 是具体的产品类,分别实现了 use() 方法。
  3. Factory 是工厂类,提供了一个静态方法 createProduct(),根据传入的参数决定创建哪种产品。
  4. main() 函数中,通过工厂类创建了不同的产品,并调用了它们的 use() 方法。

工厂方法模式:

工厂方法模式是对简单工厂模式的进一步抽象,它将工厂类抽象为一个接口,具体的工厂子类负责创建具体的产品。

#include <iostream>
#include <memory>// 产品基类
class Product {
public:virtual void use() = 0;virtual ~Product() = default;
};// 具体产品A
class ProductA : public Product {
public:void use() override {std::cout << "Using Product A" << std::endl;}
};// 具体产品B
class ProductB : public Product {
public:void use() override {std::cout << "Using Product B" << std::endl;}
};// 工厂基类
class Factory {
public:virtual std::unique_ptr<Product> createProduct() = 0;virtual ~Factory() = default;
};// 具体工厂A
class FactoryA : public Factory {
public:std::unique_ptr<Product> createProduct() override {return std::make_unique<ProductA>();}
};// 具体工厂B
class FactoryB : public Factory {
public:std::unique_ptr<Product> createProduct() override {return std::make_unique<ProductB>();}
};int main() {// 使用工厂A创建产品Astd::unique_ptr<Factory> factoryA = std::make_unique<FactoryA>();std::unique_ptr<Product> productA = factoryA->createProduct();productA->use();// 使用工厂B创建产品Bstd::unique_ptr<Factory> factoryB = std::make_unique<FactoryB>();std::unique_ptr<Product> productB = factoryB->createProduct();productB->use();return 0;
}

代码说明:

  1. Factory 是工厂基类,定义了一个纯虚函数 createProduct(),具体的工厂子类需要实现这个方法来创建具体的产品。
  2. FactoryAFactoryB 是具体的工厂类,分别负责创建 ProductAProductB
  3. main() 函数中,通过具体的工厂类创建了不同的产品,并调用了它们的 use() 方法。

抽象工厂模式:

抽象工厂模式是工厂方法模式的扩展,它提供了一个接口用于创建一系列相关或依赖的对象,而无需指定它们的具体类。

#include <iostream>
#include <memory>// 产品A基类
class ProductA {
public:virtual void use() = 0;virtual ~ProductA() = default;
};// 具体产品A1
class ProductA1 : public ProductA {
public:void use() override {std::cout << "Using Product A1" << std::endl;}
};// 具体产品A2
class ProductA2 : public ProductA {
public:void use() override {std::cout << "Using Product A2" << std::endl;}
};// 产品B基类
class ProductB {
public:virtual void use() = 0;virtual ~ProductB() = default;
};// 具体产品B1
class ProductB1 : public ProductB {
public:void use() override {std::cout << "Using Product B1" << std::endl;}
};// 具体产品B2
class ProductB2 : public ProductB {
public:void use() override {std::cout << "Using Product B2" << std::endl;}
};// 抽象工厂基类
class AbstractFactory {
public:virtual std::unique_ptr<ProductA> createProductA() = 0;virtual std::unique_ptr<ProductB> createProductB() = 0;virtual ~AbstractFactory() = default;
};// 具体工厂1
class ConcreteFactory1 : public AbstractFactory {
public:std::unique_ptr<ProductA> createProductA() override {return std::make_unique<ProductA1>();}std::unique_ptr<ProductB> createProductB() override {return std::make_unique<ProductB1>();}
};// 具体工厂2
class ConcreteFactory2 : public AbstractFactory {
public:std::unique_ptr<ProductA> createProductA() override {return std::make_unique<ProductA2>();}std::unique_ptr<ProductB> createProductB() override {return std::make_unique<ProductB2>();}
};int main() {// 使用具体工厂1创建产品A1和B1std::unique_ptr<AbstractFactory> factory1 = std::make_unique<ConcreteFactory1>();std::unique_ptr<ProductA> productA1 = factory1->createProductA();std::unique_ptr<ProductB> productB1 = factory1->createProductB();productA1->use();productB1->use();// 使用具体工厂2创建产品A2和B2std::unique_ptr<AbstractFactory> factory2 = std::make_unique<ConcreteFactory2>();std::unique_ptr<ProductA> productA2 = factory2->createProductA();std::unique_ptr<ProductB> productB2 = factory2->createProductB();productA2->use();productB2->use();return 0;
}

代码说明:

  1. ProductAProductB 是产品基类,定义了产品的接口 use()
  2. ProductA1, ProductA2, ProductB1, ProductB2 是具体的产品类,分别实现了 use() 方法。
  3. AbstractFactory 是抽象工厂基类,定义了两个纯虚函数 createProductA()createProductB(),具体的工厂子类需要实现这些方法来创建具体的产品。
  4. ConcreteFactory1ConcreteFactory2 是具体的工厂类,分别负责创建 ProductA1, ProductB1ProductA2, ProductB2
  5. main() 函数中,通过具体的工厂类创建了不同的产品,并调用了它们的 use() 方法。

这些示例展示了如何使用C++实现工厂模式的三种变体:简单工厂模式、工厂方法模式和抽象工厂模式。每种模式都有其适用的场景,选择哪种模式取决于具体的需求和设计目标。

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

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

相关文章

Suno的对手Luno:AI音乐开发「上传参考音频 - 方式二:通过URL的方式」 —— 「Luno Api系列|AI音乐API」第12篇

导读 今天来看下Luno Api的上传参考音频 - 方式一&#xff1a;通过二进制流的方式。 参考文件&#xff0c;主要是用于在创作的过程中&#xff0c;希望AI参考这个音乐的曲风和声音来进行创作&#xff0c; 这一节看看如何直接使用url的方式进行实现。 申请和使用 「已经有API…

【开源+代码解读】Search-R1:基于强化学习的检索增强大语言模型框架3小时即可打造个人AI-search

大语言模型(LLMs)在处理复杂推理和实时信息检索时面临两大挑战:知识局限性(无法获取最新外部知识)和检索灵活性不足(传统方法依赖固定检索流程)。现有方法如检索增强生成(RAG)和工具调用(Tool-Use)存在以下问题: RAG:单轮检索导致上下文不足,无法适应多轮交互场景…

Blender-MCP服务源码2-依赖分析

Blender-MCP服务源码2-依赖分析 有个大佬做了一个Blender-MCP源码&#xff0c;第一次提交代码是【2025年3月7号】今天是【2025年月15日】也就是刚过去一周的时间&#xff0c;所以想从0开始学习这个代码&#xff0c;了解一下大佬们的开发思路 1-核心知识点 from mcp.server.fas…

【孟德尔随机化】Leave-one-out analysis的异常点,判断

下面Leave-one-out analysis的结果&#xff0c;第一条线代表去掉rs174564的结果&#xff0c;一些文献把这种情况判断为异常点/离群点&#xff0c;我们接下来看看其他结果 散点图的结果&#xff0c;最旁边的就是rs174564&#xff0c;这个SNP的点 在看下RadialMR的结果&#xff0…

【计算机网络】2物理层

物理层任务:实现相邻节点之间比特(或)的传输 1.通信基础 1.1.基本概念 1.1.1.信源,信宿,信道,数据,信号 数据通信系统主要划分为信源、信道、信宿三部分。 信源:产生和发送数据的源头。 信宿:接收数据的终点。 信道:信号的传输介质。 数据和信号都有模拟或数字…

kubernetes|云原生|部署单master的kubernetes 1.25.5版本集群完全记录(使用contained 运行时)

一、 部署目标&#xff1a; kubernetes版本1.19&#xff0c;1.23的前后差异还是比较巨大的&#xff0c;到1.25版本&#xff0c;为了追求高性能&#xff0c;自然还是需要使用containerd&#xff0c;本文将主要讲述在centos7虚拟机下部署kubernetes 1.25.5集群&#xff0c;使用…

DeepSeek+Dify本地部署私有化知识库

1.Windows安装docker Windows安装Docker-CSDN博客 2.安装olloma https://ollama.com/ 安装完成&#xff0c;可以在桌面右下角看到olloma图标 3.安装deepseekR1模型 ollama官网&#xff08;deepseek-r1&#xff09;&#xff0c;找到deepseek模型 选择合适大小的模型&#xff…

[Linux][经验总结]Ubuntu6.11.0 docker更换镜像源(实操可用的正确方法)

一、前言 关于Ubuntu更换docker镜像源&#xff0c;网上有很多的教程&#xff0c;但在实操中发现&#xff0c;更换的源无法生效——原因是我的docker是在系统安装时&#xff0c;选择附加安装的package的方式安装的。 现将处理过程记录如下。 二、获取镜像源 在网上随便找个几…

NHANES指标推荐:BRI!

文章题目&#xff1a;Association of body roundness index with cardiovascular disease in patients with cardiometabolic syndrome: a cross-sectional study based on NHANES 2009-2018 DOI&#xff1a;10.3389/fendo.2025.1524352 中文标题&#xff1a;心脏代谢综合征患者…

3.水中看月

前言 这篇文章讲解套接字分配IP地址和端口号。这部分内容也相对有些枯燥&#xff0c;但并不难&#xff0c;而 且是学习后续那些有趣内容必备的基础知识&#xff08;计算机网络基础&#xff09;。 一、分配给套接字的IP地址与端口号 IP是InternetProtocol&#xff08;网络协议…

Linux驱动开发-①pinctrl 和 gpio 子系统②并发和竞争③内核定时器

Linux驱动开发-①pinctrl 和 gpio 子系统②并发和竞争③内核定时器 一&#xff0c;pinctrl 和 gpio 子系统1.pinctrl子系统2.GPIO子系统 二&#xff0c;并发和竞争1.原子操作2.自旋锁3.信号量4.互斥体 三&#xff0c;按键实验四&#xff0c;内核定时器1.关于定时器的有关概念1.…

奇安信二面

《网安面试指南》https://mp.weixin.qq.com/s/RIVYDmxI9g_TgGrpbdDKtA?token1860256701&langzh_CN 5000篇网安资料库https://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247486065&idx2&snb30ade8200e842743339d428f414475e&chksmc0e4732df793fa3bf39…

Python库安装报错解决思路以及机器学习环境配置详细方案

文章目录 概要第三方库gdalpymoltalibmahotasgraphviznltk-datalazypredictscikit-surprisenb_extensionspyqt5-toolsspacy、en_core_web_sm 机器学习GPU-torch安装torch_geometric安装ubuntu安装显卡驱动dlib安装torch-cluster、torch-scatter、torch-sparse和torch-geometric…

Power Apps 技术分享:连接SharePoint列表数据源

前言 在使用Power Apps的时候&#xff0c;使用列表作为数据源是非常方便和经济的&#xff0c;列表创建简单&#xff0c;SharePoint的存储也不像Dataverse需要按照容量付费。 正文 1.我们先在SharePoint中建一个列表&#xff0c;添加一些测试数据&#xff0c;如下图&#xff1a;…

【Linux】learning notes(4)cat、more、less、head、tail、vi、vim

文章目录 catmore 查看整个文件less 查看整个文件head 查看部分文件tail 查看部分文件vim / vi cat cat 命令在 Linux 和 Unix 系统中非常常用&#xff0c;它用于连接文件并打印到标准输出设备&#xff08;通常是屏幕&#xff09;。虽然 cat 的基本用法很简单&#xff0c;但它…

C++11函数包装器

目录 std::function 注意事项 包装静态成员函数 包装非静态成员函数 std::bind 用法 应用场景 std::function function是C11引入的类&#xff0c;可以用任何可调用对象作为参数&#xff0c;构造出一个新对象。 可调用对象有函数指针&#xff0c;仿函数&#xff0c;lamb…

maven的安装配置

目录 一、官网下载压缩包 二、配置环境变量 设置 MAVEN_HOME 添加 MAVEN_HOME\bin 到 PATH 三、配置本机仓库和远程仓库 四、配置idea 一、官网下载压缩包 Download Apache Maven – Maven 如上图。选择这个压缩包 选择好文件&#xff0c;下载完后&#xff0c;配置环境变…

分布式事务

1 事务 众所周知&#xff0c;事务具有ACID四大特性&#xff1a; 原子性&#xff08;Atomicity&#xff09;&#xff1a;事务作为一个整体被执行&#xff0c;包含在其中的对数据库的操作要么全部被执行&#xff0c;要么都不执行。 一致性&#xff08;Consistency&#xff09;&a…

Postman中Authorization和Headers的区别

案例 笔者在进行token验证的时候碰到的问题 一般如果是进行token验证&#xff0c;大部分是在Headers下面添加token名称及token的值 这样&#xff1a;后端提取请求头的token即可 还有一种是&#xff0c;左侧选择Bearer Token&#xff0c;右侧添加token的值,后端传递的 大概…

1.备战SISAP 2025挑战:调研2024挑战

简介 紧张刺激的SISAP 2025 challenge发布了&#xff0c;此博客用于记录备战的一些准备&#xff0c;思路和实验。 25年挑战介绍 详细信息参考SISAP Indexing challenge 2025 Task 1&#xff1a;内存受限索引 这项任务要求参与者开发具有reranking&#xff08;重排&#xf…