简单工厂,工厂方法,抽象工厂模式

文章目录

  • 简单工厂模式
    • 简单工厂模式的角色
    • 简单工厂的介绍
  • 工厂方法
    • UML图片
    • 代码
    • 工厂方法总结:
  • 抽象工厂
    • 代码实现

简单工厂模式

简单工厂模式(Simple Factory Pattern)专门定义一个类来负责创建其他类的实例,被创建的实例通常具有共同的父类。

简单工厂模式,是一种实例化对象的方式,只要输入需要实例化对象的名字,就可以通过工厂对象的相应工厂函数来制造你需要的对象。

简单工厂模式的角色

(1)Factory工厂角色(工厂类):

工厂角色即工厂类,是简单工厂模式的核心,负责创建所有实例的内部逻辑,工厂类可以被外界直接调用,创建所需要的产品对象。

(2)Product(抽象产品角色):

抽象产品角色是简单工厂模式所创建的所有对象的父类,负责描述所有实例所共有的公告接口。所创建的具体产品对象都是其子类对象。

(3)ConcreteProduct(具体产品角色):

具体产品角色是简单工厂模式的创建目标。每个具体产品角色都继承了抽象产品角色,需要实现定义在抽象产品中的方法。

简单工厂的介绍

简单工厂(Simple Factory)不属于标准的OOP设计模式中的一项,在编写大型C++软件的时候,代码里面会出现很多的类,每次创建对象的时候,都需要通过new 类名称的方式来生成对象,这样一来,用户需要记忆很多类的名称,暂且不管记不记得住,这样的设计使得代码很难维护,类名如果做了改变,那么所有使用类名称的地方都需要去修改,耦合性太强,不符合我们软件设计的思想,Simple Factory就是在这样的需求下诞生的。

在这里插入图片描述

我们把对象的创建全部都封装在了SimpleFactory的create方法中,在这里我们传入一个Product就可以创建一个对象,省略了new的过程

#include <iostream>
#include <string>
using namespace std;
//产品类型的声明,抽象产品
class  Product
{
public:
	Product(string name) : ProductName(name) {};
	virtual void show() = 0;
protected:
	string ProductName;
};

//具体产品的声明
class BaoMa :public Product
{
public:
	BaoMa(string name) : Product(name) {};
	void show()
	{
		std::cout << "This is BaoMa Product" << char(10);
		std::cout << ProductName << char(10);
	}
};


//具体产品2 的申明
class BengChi :public Product
{
public:
	BengChi(string name) : Product(name) {};
	void show()
	{
		std::cout << "This is <BengChi> Product" << char(10);
		std::cout << ProductName << char(10);
	}
};

enum ProductType
{
	 BWM,
	 BC
};

class SimpleFactory
{
public:
	//通过传入的枚举类型,创建对应的对象,返回对象的基类指针
	Product* createProduct(ProductType type)
	{
		switch (type)
		{
		case BWM:
			return new BaoMa("宝马");
			break;
		case BC:
			return new BengChi("奔驰");
			break;
		default:
			std::cout << "你输入了错误的名字" << type << char(10);
			break;
		}
	}
};


int main()
{
	//创建简单工厂的实例
	SimpleFactory sf;
	unique_ptr<Product> p1(sf.createProduct(BWM));
	unique_ptr<Product> p2(sf.createProduct(BC));


	p1->show();
	p2->show();

	return 0;
}

工厂方法

UML图片

在这里插入图片描述

通过产生具体的工厂创建具体的产品,做到了在扩充新产品时,能够达到软件设计的“开-闭”原则

代码

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

class AbstractProduct
{
public:
	AbstractProduct(string name): _name(name){}
	//模拟产品对象的抽象方法
	virtual void show() = 0;
protected:
	string _name;
};

class ProductA:public AbstractProduct
{
public:
	//其实这是一个委托构造函数
	ProductA(string name) : AbstractProduct(name) {}
	void show() 
	{ 
		std::cout << "这是一个产品A的对象"; char(10);
		std::cout << "string name : " << _name << char(10);
	}
};

class ProductB :public AbstractProduct
{
public:
	//其实这是一个委托构造函数
	ProductB(string name) : AbstractProduct(name) {}
	void show() { 
		std::cout << "这是一个产品B的对象"<< char(10);
		std::cout << "string name : " << _name << char(10);
	}
};

//定义枚举类型的数据
enum  ProductType
{
	XIAOMI,
	HUAWEI,
};

//定义抽象工厂
class AbstractFactory
{
public:
	virtual AbstractProduct* createProduct() = 0;
};

//小米手机工厂
class XiaoMiFactory:public AbstractFactory
{
public:
	AbstractProduct* createProduct()
	{
		std::cout << "调用了小米工厂的创建方法" << char(10);
		return new ProductA("小米手机");
	}
};

//华为手机工厂
class HuaWeiFactory :public AbstractFactory
{
public:
	AbstractProduct* createProduct()
	{
		std::cout << "调用了华为工厂的创建方法" << char(10);
		return new ProductB("华为手机");
	}
};

int main()
{
	//在这里我们使用智能指针,创建具体工厂
	unique_ptr<AbstractFactory> Fac1(new XiaoMiFactory);
	unique_ptr<AbstractFactory> Fac2(new HuaWeiFactory);

	//通过工厂方法创建产品
	unique_ptr<AbstractProduct> Pro1(Fac1->createProduct());
	unique_ptr<AbstractProduct> Pro2(Fac2->createProduct());

	Pro1->show();
	Pro2->show();

	return 0;
}

工厂方法可以解决一簇产品(一款产品有多种形态)的问题,并且工厂一次只能生产一种产品;但需要一个工厂能够同时生产多款产品时,则需要用到抽象工厂模式

工厂方法总结:

仔细理解上面的工厂方法模式,会发现一个问题,就是每一个实例工厂负责生产一个实例产品,也就是一个产品对应一个工厂,一个工厂对应一个产品,那么小米不仅仅生产手机,还生产耳机,智能手环,智能插座等等相关的小米产品簇,不可能给这每一个产品都创建一个工厂类,那样的话代码中的类就太多了,不好维护,而且也不符合实际情况。

实际上小米或者华为的工厂里面,有相关联的产品簇都是在一个工厂完成创建的;BMW或者Audi汽车制造工厂除了生产汽车,生产线上也有可能生产轮胎,或者其它的汽车附属产品。

所以对于包含产品簇这么一类实体关系的设计,就需要使用Abstract Factory抽象工厂了,你也可以把上面的工厂方法看作只生产一种产品的抽象工厂,本质是相同的。

抽象工厂

在这里插入图片描述

可以看到,抽象工厂模式把一个产品簇的产品放在一个工厂类中去创建,不仅大大减少了工厂类的个数,更符合现实中工厂生产产品的模式。根据上面的内容描述,仔细思考简单工厂,工厂方法和抽象工厂的区别联系

代码实现

#include <iostream>
using namespace std;

//抽象产品 手机类
class AbstractPhone
{
public:
	AbstractPhone(string name) : _name(name) {};
	//模拟产品对象的一个抽象的方法
	virtual void show() = 0; 
protected:
	string _name;
};

//产品实体类型 小米
class XiaoMiPhone: public AbstractPhone
{
public:
	XiaoMiPhone(string name) : AbstractPhone(name) {};
	void show()
	{
		std::cout << "这是" << _name << "手机" << char(10);
	}
};

//产品实体类型 华为
class HuaweiPhone : public AbstractPhone
{
public:
	HuaweiPhone(string name) : AbstractPhone(name) {};
	void show()
	{
		std::cout << "这是" << _name << "手机" << char(10);
	}
};

//抽象产品 手环
class AbstractCircle
{
public:
	AbstractCircle(string name) : _name(name) {};
	//模拟产品对象的一个抽象的方法
	virtual void show() = 0;
protected:
	string _name;
};

//产品实体 小米手环
class  XiaoMiCircle:public AbstractCircle
{
public:
	XiaoMiCircle(string name) :AbstractCircle(name) {};
	void show()
	{
		std::cout << "这是" << _name << "手环" << char(10);
	}
};

//产品实体 华为手环
class  HuaWeiCircle :public AbstractCircle
{
public:
	HuaWeiCircle(string name) :AbstractCircle(name) {};
	void show()
	{
		std::cout << "这是" << _name << "手环" << char(10);
	}
};

//抽象工厂,创建一个产品族的设备的产品
class AbstractFactory
{
public:
	//创建手机的纯虚函数
	virtual AbstractPhone* createPhone() = 0;
	virtual AbstractCircle* createCircle() = 0;
};

//生产小米的工厂
class XiaoMiFactory:public AbstractFactory
{
public:
	virtual AbstractPhone* createPhone()
	{
		return new XiaoMiPhone("小米14Pro");
	}
	virtual AbstractCircle* createCircle()
	{
		return new XiaoMiCircle("小米手环NFC版本");
	}
};

//生产华为的工厂
class HuaWeiFactory :public AbstractFactory
{
public:
	virtual AbstractPhone* createPhone()
	{
		return new XiaoMiPhone("华为Mate60 pro");
	}
	virtual AbstractCircle* createCircle()
	{
		return new XiaoMiCircle("华为GT Runing");
	}
};



int main()
{
	//利用智能指针,创建生产的工厂
	unique_ptr< AbstractFactory> fac1(new XiaoMiFactory);
	unique_ptr< AbstractFactory> fac2(new HuaWeiFactory);

	//通过工厂来生产手机的
	unique_ptr< AbstractPhone > phone1(fac1->createPhone());
	unique_ptr< AbstractPhone > phone2(fac2->createPhone());
	phone1->show();
	phone2->show();

	//生产手环
	unique_ptr< AbstractCircle > cir1(fac1->createCircle());
	unique_ptr< AbstractCircle > cir2(fac2->createCircle());
	cir1->show();
	cir2->show();

	return 0;
}

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

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

相关文章

商品推荐系统+可视化+2种协同过滤推荐算法 Django框架 大数据毕业设计(附源码+论文)✅

毕业设计&#xff1a;2023-2024年计算机专业毕业设计选题汇总&#xff08;建议收藏&#xff09; 毕业设计&#xff1a;2023-2024年最新最全计算机专业毕设选题推荐汇总 &#x1f345;感兴趣的可以先收藏起来&#xff0c;点赞、关注不迷路&#xff0c;大家在毕设选题&#xff…

rime中州韵小狼毫 inputShow lua Translator 输入字符透传翻译器

在 rime中州韵小狼毫 help lua Translator 中我们分享了如何使用 lua 脚本定义一个 translator&#xff0c;并以 五笔・拼音 为例引用了该 translator&#xff0c;并且达到了预期的效果。 今天&#xff0c;我们继续通过 lua 脚本为 rime中州韵/小狼毫 输入法打造一个 translat…

std::setlocale详解

头文件 #include <clocale>作用 std::setlocale是C标准库中的一个函数&#xff0c;用于设置当前程序的本地化&#xff08;locale&#xff09;环境。 setlocale 函数安装指定的系统本地环境或其一部分&#xff0c;作为新的 C 本地环境。 修改保持效果&#xff0c;并影…

【MyBatis】配置 SQL 提示(IDEA 代码自动补全)和解决未能解析数据库 Unable to resolve table ‘employee‘ 问题

默认在 MyBatis 中编写 SQL 语句是不识别的。 需要做如下配置&#xff1a; 然后 SQL 语句可以出现提示&#xff0c;但是自己的数据库会报错&#xff1a; 原因是 IDEA 没有和 数据库 建立连接&#xff0c;需要连接数据库&#xff08;注意&#xff1a;需要具体到某个 Schema&…

人工智能_机器学习080_KMeans聚类算法原理和流程_KMeans损失函数_随机聚类中心_对异常值_初始值敏感---人工智能工作笔记0120

然后我们来看一下KMeans聚类算法的原理 可以看到,聚类的过程是,首先1.将N个样本数据,映射到K个簇中,这个K就是我们指定的划分成几类,比如3对吧,那么 指定了K个类别以后,然后 我们再去随机指定K个,簇中心 然后再去计算,我们指定的这个簇中心,到簇中所有样本的距离的平均值, 我…

多模态——旷视大模型Vary更细粒度的视觉感知实现文档级OCR或图表理解

概述 现代大型视觉语言模型&#xff08;LVLMs&#xff09;&#xff0c;例如CLIP&#xff0c;使用一个共同的视觉词汇&#xff0c;以适应多样的视觉任务。然而&#xff0c;在处理一些需要更精细和密集视觉感知的特殊任务时&#xff0c;例如文档级OCR或图表理解&#xff0c;尤其…

2024年测试岗最新自动化测试面试题整理,干货满满

一、接口测试 1、接口测试用例的编写要点有哪些&#xff1f;&#xff08;问法二&#xff1a;接口测试用例设计需要考虑哪些方面&#xff1f;问法三&#xff1a;接口测试中有哪些要注意的测试点&#xff1f;&#xff09; 1&#xff09; 考虑输入参数和输出参数的合法性&#x…

B01、类加载子系统-02

JVM架构图-英文版 中文版见下图&#xff1a; 1、概述类的加载器及类加载过程 1.1、类加载子系统的作用 类加载器子系统负责从文件系统或者网络中加载Class文件,class文件在文件开头有特定的文件标识。ClassLoader只负责class文件的加载,至于它是否可以运行,则由Execution Engi…

4-文献阅读-A Data-driven Base Station Sleeping Strategy Based on Traffic Prediction

目录 文献阅读—A Data-driven Base Station Sleeping Strategy Based on Traffic Prediction0、选这篇文章的原因1、文章的主要内容和贡献2、使用的数据集3、结果及分析4、郭郭有话说 文献阅读—A Data-driven Base Station Sleeping Strategy Based on Traffic Prediction 0…

Unity中URP下的添加雾效支持

文章目录 前言一、URP下Shader支持雾效的步骤1、添加雾效变体2、在Varying结构体中添加雾效因子3、在顶点着色器中&#xff0c;我们使用内置函数得到雾效因子4、在片元着色器中&#xff0c;把输出颜色 和 雾效因子混合输出 二、在Unity中打开雾效三、测试代码 前言 我们使用之…

Python学习笔记之(一)搭建Python 环境

搭建Python 环境 1. 使用工具准备1.1 Python 安装1.1.1 下载Python 安装包1.1.2 安装Python 1.2 VScode 安装1.2.1 下载VScode安装包1.2.2 给VScode安装Python 扩展 2. 第一次编写Python 程序 本篇文章以Windows 系统为例。 1. 使用工具准备 1.1 Python 安装 1.1.1 下载Pytho…

【JavaSE】string与StringBuilder和StringBuffer

区别&#xff1a; 不可变性&#xff1a; String&#xff1a; String 类是不可变的&#xff0c;一旦创建就不能被修改。对字符串的任何操作都会创建一个新的字符串对象。StringBuffer&#xff1a; StringBuffer 是可变的&#xff0c;允许对字符串进行修改&#xff0c;而不创建新…

案例089:基于微信小程序的校园综合服务平台设计与实现

文末获取源码 开发语言&#xff1a;Java 框架&#xff1a;SSM JDK版本&#xff1a;JDK1.8 数据库&#xff1a;mysql 5.7 开发软件&#xff1a;eclipse/myeclipse/idea Maven包&#xff1a;Maven3.5.4 小程序框架&#xff1a;uniapp 小程序开发软件&#xff1a;HBuilder X 小程序…

李沐机器学习系列1--- 线性规划

1 Introduction 1.1 线性回归函数 典型的线性回归函数 f ( x ) w ⃗ ⋅ x ⃗ f(x)\vec{w} \cdot \vec{x} f(x)w ⋅x 现实生活中&#xff0c;简单的线性回归问题很少&#xff0c;这里有一个简单的线性回归问题。房子的价格和房子的面积以及房子的年龄假设成线性关系。 p r …

Java项目:107SpringBoot房屋租赁网站

博主主页&#xff1a;Java旅途 简介&#xff1a;分享计算机知识、学习路线、系统源码及教程 文末获取源码 一、项目介绍 房屋租赁网站基于SpringBootMybatis开发&#xff0c;系统分为管理员和普通用户两种角色。 管理员功能如下&#xff1a; 登录修改密码查看用户房屋管理图…

内联函数的作用

目的 主要为了提升程序运行速度。 分析 当程序调用一个函数时&#xff0c;程序暂停执行当前指令&#xff0c;跳到函数体处执行&#xff0c;在函数执行完后&#xff0c;返回原来的位置继续执行。如果该函数为内联函数&#xff0c;则不同跳&#xff0c;是因为该内联函数直接插…

java每日一题——双色球系统(答案及编程思路)

前言&#xff1a; 打好基础&#xff0c;daydayup! 题目&#xff1a;要求如下&#xff08;同时&#xff1a;红球每个号码不可以相同&#xff09; 编程思路&#xff1a;1&#xff0c;创建一个可以录入数字的数组&#xff1b;2&#xff0c;生成一个可以随机生成数字的数组&#xf…

顶帽运算在OpenCv中的应用

项目背景 假如我们拍了一张自拍&#xff0c;想为自己的照片添加一个酷炫的火星飞舞的效果&#xff0c;素材库中正好有一张火焰的照片&#xff0c;如果想去除图中的火焰&#xff0c;只保留火星效果&#xff0c;可以使用顶帽子算法 图片中的火星部分正好属于比周围亮一些的斑块…

灰度发布及声明式资源管理(yaml文件)

一、三种常见的项目发布方式 1&#xff09;蓝绿发布 2&#xff09;灰度发布【常用】 3&#xff09;滚动发布 应用程序升级&#xff0c;面临最大的问题是新旧业务之间的切换 立项-定稿-需求发布-开发-测试-发布&#xff0c;测试上线后&#xff0c;再完美也会有问题&#xff0c;为…

Windows下Jenkins自动化部署SpringBoot应用

Windows下Jenkins自动化部署SpringBoot应用 1、下载安装包 下载地址&#xff1a; 一个是 msi 程序&#xff1a; https://mirrors.aliyun.com/jenkins/windows/ 一个是 war 程序&#xff1a; https://get.jenkins.io/war-stable/ https://mirrors.jenkins.io/war/ 这里我…