设计模式:简单工厂模式(Simple Factory)

news2024/10/6 20:29:22

设计模式:简单工厂模式(Simple Factory)

  • 设计模式:简单工厂模式(Simple Factory)
    • 模式动机
    • 模式定义
    • 模式结构
    • 时序图
    • 模式实现
    • 测试
    • 模式分析
    • 实例:Qt 控件类
    • 优缺点
    • 适用环境
    • 模式应用

设计模式:简单工厂模式(Simple Factory)

简单工厂模式(Simple Factory)属于创建型模式(Creational Pattern)的一种。

创建型模式(Creational Pattern)对类的实例化过程进行了抽象,能够将软件模块中对象的创建和对象的使用分离。为了使软件的结构更加清晰,外界对于这些对象只需要知道它们共同的接口,而不清楚其具体的实现细节,使整个系统的设计更加符合单一职责原则。

创建型模式在创建什么(What),由谁创建(Who),何时创建(When)等方面都为软件设计者提供了尽可能大的灵活性。创建型模式隐藏了类的实例的创建细节,通过隐藏对象如何被创建和组合在一起达到使整个系统独立的目的。

模式动机

意图:定义一个创建对象的接口,让其子类自己决定实例化哪一个工厂类,工厂模式使其创建过程延迟到子类进行。

考虑一个简单的软件应用场景,一个软件系统可以提供多个外观不同的按钮(如圆形按钮、矩形按钮、菱形按钮等), 这些按钮都源自同一个基类,不过在继承基类后不同的子类修改了部分属性从而使得它们可以呈现不同的外观,如果我们希望在使用这些按钮时,不需要知道这些具体按钮类的名字,只需要知道表示该按钮类的一个参数,并提供一个调用方便的方法,把该参数传入方法即可返回一个相应的按钮对象,此时,就可以使用简单工厂模式。

模式定义

简单工厂模式(Simple Factory)又称为静态工厂方法(Static Factory Method)模式。

在简单工厂模式中,可以根据参数的不同返回不同类的实例。简单工厂模式专门定义一个类来负责创建其他类的实例,被创建的实例通常都具有共同的父类。

模式结构

简单工厂模式(Simple Factory)包含以下角色:

  1. 工厂(Factory):工厂角色负责实现创建所有实例的内部逻辑。
  2. 抽象产品(Abstract Product):定义了产品的共同接口或抽象类。它可以是具体产品类的父类或接口,规定了产品对象的共同方法。
  3. 具体产品(Concrete Product):实现了抽象产品接口,定义了具体产品的特定行为和属性。具体产品是创建目标,所有创建的对象都充当这个角色的某个具体类的实例。

在这里插入图片描述

时序图

在这里插入图片描述

模式实现

简单工厂类 SimpleFactory.h:

#ifndef _SIMPLE_FACTORY_H_
#define _SIMPLE_FACTORY_H_

#include <string>
#include "ConcreteProductA.h"
#include "ConcreteProductB.h"

class SimpleFactory
{
public:
	static AbstractProduct* createProduct(const std::string& productType)
	{
		if (productType == "A")
			return new ConcreteProductA();
		else if (productType == "B")
			return new ConcreteProductB();
		return nullptr;
	}
};

#endif // !_SIMPLE_FACTORY_H_

抽象产品类 AbstractProduct.h:

#ifndef _ABSTRACT_PRODUCT_H_
#define _ABSTRACT_PRODUCT_H_

#include <iostream>

class AbstractProduct
{
public:
	virtual void use() = 0;
};

#endif // !_ABSTRACT_PRODUCT_H_

具体产品A类 ConcreteProductA.h:

#ifndef _CONCRETE_PRODUCT_A_H_
#define _CONCRETE_PRODUCT_A_H_

#include "AbstractProduct.h"

class ConcreteProductA : public AbstractProduct
{
public:
	void use() override
	{
		std::cout << "Using Concrete Product A" << std::endl;
	}
};

#endif // !_CONCRETE_PRODUCT_A_H_

具体产品B类 ConcreteProductB.h:

#ifndef _CONCRETE_PRODUCT_B_H_
#define _CONCRETE_PRODUCT_B_H_

#include "AbstractProduct.h"

class ConcreteProductB : public AbstractProduct
{
public:
	void use() override
	{
		std::cout << "Using Concrete Product B" << std::endl;
	}
};

#endif // !_CONCRETE_PRODUCT_B_H_

测试

测试代码 main.cpp:

#include <stdlib.h>
#include "SimpleFactory.h"
#include "ConcreteProductA.h"
#include "ConcreteProductB.h"

int main()
{
	SimpleFactory simpleFactory;
	AbstractProduct* productA = simpleFactory.createProduct("A");
	AbstractProduct* productB = simpleFactory.createProduct("B");

	if (productA)
		productA->use();
	if (productB)
		productB->use();
	
	delete productA;
	delete productB;

	system("pause");
	return 0;
}

运行结果:

在这里插入图片描述

模式分析

  • 将对象的创建和对象本身业务处理分离可以降低系统的耦合度,使得两者修改起来都相对容易。
  • 在调用工厂类的工厂方法时,由于工厂方法是静态方法,使用起来很方便,可通过类名直接调用,而且只需要传入一个简单的参数即可,在实际开发中,还可以在调用时将所传入的参数保存在XML等格式的配置文件中,修改参数时无须修改任何源代码。
  • 简单工厂模式最大的问题在于工厂类的职责相对过重,增加新的产品需要修改工厂类的判断逻辑,这一点与开闭原则是相违背的。
  • 简单工厂模式的要点在于:当你需要什么,只需要传入一个正确的参数,就可以获取你所需要的对象,而无须知道其创建细节。

实例:Qt 控件类

在开发一个用户界面过程中,我需要三种控件类型,分别是按钮、图像和文本。这些控件都有一些工厂的特点,比如都有位置属性(位于界面的哪个地方)、大小属性(长度和宽度)等等。因此创建一个Item(参考Qt)作为它们的父类,Button(按钮)、Image(图像)和Text(文本)都继承于它。

下面是参考代码:

#include <iostream>  
#include <memory> 
#include <string> 
#include <stdlib.h>
using namespace std;

// Item 父类
class Item
{
protected:
	int m_width;
	int m_height;

public:
	Item(int width, int height) : m_width(width), m_height(height) {}
	virtual ~Item() {}
};

// Button 子类
class Button : public Item
{
public:
	Button(int width, int heght) : Item(width, heght)
	{
		cout << "用户界面创建了一个按钮" << endl;
	}
};

// Image 子类
class Image : public Item
{
public:
	Image(int width, int heght) : Item(width, heght)
	{
		cout << "用户界面创建了一个图像" << endl;
	}
};

// Text 子类
class Text : public Item
{
public:
	Text(int width, int heght) : Item(width, heght)
	{
		cout << "用户界面创建了一个文本" << endl;
	}
};

// ItemFactory 工厂类
class ItemFactory
{
public:
	static unique_ptr<Item> createItem(string itemType)
	{
		if (itemType == "button")
		{
			// newItem = new Button(50, 50);
			return make_unique<Button>(50, 50);
		}
		else if (itemType == "image")
		{
			// newItem = new Image(100, 100);
			return make_unique<Image>(100, 100);
		}
		else if (itemType == "text")
		{
			// newItem = new Text(100, 30);
			return make_unique<Text>(100, 30);
		}
		return NULL;
	}
};

int main()
{
	ItemFactory itemFac;
    auto pI1 = itemFac.createItem("button"); // 创建了一个 Button
    auto pI2 = itemFac.createItem("image"); // 创建了一个 Image
    auto pI3 = itemFac.createItem("text"); // 创建了一个 Text
 
//    delete pI1;
//    delete pI2;
//    delete pI3;
		
	system("pause");
	return 0;
}

我们使用简单工厂模式后,创建各种各样的控件,就不用和它们控件本身的类打交道(Button、Image、Text),而是通过ItemFactory的成员函数createFactory,将不同字符串标识传入进去,从而返回不同的控件,这其实就是一种封装变化。

优缺点

优点:

  1. 封装性好: 简单工厂模式封装了对象创建的细节,客户端不需要直接实例化具体的产品类。这有助于隐藏具体产品类的实现细节,使得客户端代码更加简洁和易于维护。
  2. 解耦: 简单工厂模式降低了客户端与具体产品类之间的耦合度。客户端只需要与工厂类交互,而不需要知道具体产品类的实现细节。这有助于减少代码的依赖性和提高系统的可维护性。
  3. 代码复用: 工厂类集中了产品对象的创建逻辑,可以在多个地方重用该工厂类来创建对象,提高了代码的复用性。
  4. 扩展性好: 当需要添加新的产品类时,只需要在工厂类中添加相应的创建逻辑,而无需修改客户端代码。这符合开闭原则,即对扩展开放,对修改封闭。

缺点:

  1. 违反单一职责原则: 简单工厂类通常负责创建多种类型的产品对象,这可能导致工厂类的职责过多,违反了单一职责原则。当产品类型较多时,工厂类的代码可能会变得庞大而难以维护。
  2. 类型判断逻辑复杂: 简单工厂模式通常需要根据客户端提供的参数或类型信息来判断需要创建哪种类型的产品对象。随着产品类型的增加,类型判断逻辑可能会变得复杂和易错。
  3. 不易于扩展新的产品族: 简单工厂模式是针对一个产品族设计的,如果需要添加新的产品族(即一组具有共同主题的产品),可能需要修改工厂类的代码,这违反了开闭原则。
  4. 不易于使用继承等面向对象特性: 简单工厂模式由于使用了静态工厂方法,造成工厂角色无法形成基于继承的等级结构;由于简单工厂模式通常直接返回具体产品类的实例,而不是通过继承等面向对象特性来创建对象,因此可能无法充分利用面向对象编程的优势。

适用环境

C++ 简单工厂模式的应用场景主要包括以下几种情况:

  1. 创建对象不需要知道具体类的情况: 客户端既不需要关心创建细节,甚至连类名都不需要记住,只需要知道类型所对应的参数。
  2. 需要创建的对象较少且不会经常变动: 由于创建的对象较少,不会造成工厂方法中的业务逻辑太过复杂。

模式应用

  1. JDK类库中广泛使用了简单工厂模式,如工具类java.text.DateFormat,它用于格式化一个本地日期或者时间。
    public final static DateFormat getDateInstance();
    public final static DateFormat getDateInstance(int style);
    public final static DateFormat getDateInstance(int style,Locale
    locale);
    
  2. Java加密技术。
    比如,获取不同加密算法的密钥生成器:
    public final static DateFormat getDateInstance();
    public final static DateFormat getDateInstance(int style);
    public final static DateFormat getDateInstance(int style,Locale locale);
    

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

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

相关文章

李沐-26 网络中的网络 NiN【动手学深度学习v2】

主要记载关于全局平均池化层&#xff08;Global Average Pooling, GAP&#xff09;中如下两点的理解&#xff1a; 1. GAP的原理 2. 相对于全连接层&#xff0c;GAP具有更少的参数 为了直观地说明全局平均池化层相对于全连接层具有更少的参数&#xff0c;我们可以构造一个简…

博客文章:AWS re:Invent 2023 新产品深度解析 - 第四部分

TOC &#x1f308;你好呀&#xff01;我是 是Yu欸 &#x1f30c; 2024每日百字篆刻时光&#xff0c;感谢你的陪伴与支持 ~ &#x1f680; 欢迎一起踏上探险之旅&#xff0c;挖掘无限可能&#xff0c;共同成长&#xff01; 写在最前面 去年发布文章的一部分&#xff0c;由于内…

bugku-web-login2

这里提示是命令执行 抓包发现有五个报文 其中login.php中有base64加密语句 $sql"SELECT username,password FROM admin WHERE username".$username.""; if (!empty($row) && $row[password]md5($password)){ } 这里得到SQL语句的组成&#xff0c;…

SOLIDWORKS批量改名工具个人版 慧德敏学

每个文件都会有自己的名字&#xff0c;SOLIDWOKRKS模型也不例外。但是如果从资源管理器直接修改模型的文件名&#xff0c;就会导致模型关联的丢失&#xff0c;导致装配体打开之后找不到模型&#xff0c;因此就需要使用SOLIDWORKS的重命名功能。 SOLIDWORKS批量改名插件- Solid…

Blazor 下的 Json 编辑器

最近恰好碰到个比较冷门的需求&#xff0c;就是在线编码 Json&#xff0c;这其中有Json的语法着色&#xff0c;有Json对象属性数据类型的限制&#xff0c;其实要是单纯改一下Json字符串也不是难事&#xff0c;就是没法控制让用户只能给属性值&#xff0c;而不是属性名称&#x…

【随笔】Git 高级篇 -- 远程服务器拒绝 git push reset(三十二)

&#x1f48c; 所属专栏&#xff1a;【Git】 &#x1f600; 作  者&#xff1a;我是夜阑的狗&#x1f436; &#x1f680; 个人简介&#xff1a;一个正在努力学技术的CV工程师&#xff0c;专注基础和实战分享 &#xff0c;欢迎咨询&#xff01; &#x1f496; 欢迎大…

C++:仿函数模拟实现STL-priority_queue

优先级队列模拟实现 1.文档了解2.仿函数实现优先级队列仿函数1.定义2.语法3.使用 3.模拟实现源码 1.文档了解 从priority_queue的文档中&#xff0c;我们可以得出以下几点&#xff1a; 1.priority_queue是一个容器适配器 2.priority_queue它实质是一个堆&#xff0c;且默认为大…

【Leetcode】string类刷题

&#x1f525;个人主页&#xff1a;Quitecoder &#x1f525;专栏&#xff1a;Leetcode刷题 目录 1.仅反转字母2.字符串中第一个唯一字符3.验证回文串4.字符串相加5.反转字符串I I6.反转字符串中的单词III7.字符串相乘8.把字符串转换为整数 1.仅反转字母 题目链接&#xff1a;…

MySQL 8.0.19安装教程(windows 64位)

在c盘目录下的Program Files目录下创建MySQL目录&#xff0c;将下载好的mysql解压到里面 解压完是这个样子 配置初始化的my.ini文件的文件 [mysqld] # 设置3306端口 port3306 # 设置mysql的安装目录 basedirC:\Program Files\MySQL # 设置mysql数据库的数据的存放目录 datad…

便携式手提万兆网络协议测试仪

便携式手提万兆网络协议测试仪 平台简介 便携式手提万兆网络协议测试仪&#xff0c;以FPGA万兆卡和X86主板为基础&#xff0c;构建便携式的手提设备。 FPGA万兆卡是以Kintex-7XC7K325T PCIeX4的双路万兆光纤网络卡&#xff0c;支持万兆网络数据的收发和网络协议的定制设计。 …

weblogic JSP action的配置

action(如xxx.do&#xff09;可以在Java文件中通过注解的方式配置&#xff0c;也可以在web.xml中进行配置 在java文件中配置的场合 WebServlet(xxxx.do) 并实现支持的方法&#xff1a;doGet或doPost等 或者 WebServlet(xxxx.do) 并实现service方法 所有method的处理方法都会…

OpenHarmony音频和音乐编码格式—vorbis

简介 一种通用音频和音乐编码格式。 Vorbis编解码器规范属于公共领域。所有技术细节都已发布并记录&#xff0c;任何软件实体都可以充分利用该格式&#xff0c;而无需支付许可费、版税或专利问题。 下载安装 直接在OpenHarmony-SIG仓中搜索vorbis并下载。 使用说明 以OpenHa…

常用UI组件

一、文本组件 1.1 概述 Text为文本组件&#xff0c;用于显示文字内容 1.2 参数 Text组件的参数类型为string | Resource Entry Component struct Index {build() {Column({space : 50}) {Text(你好).fontSize(50)}.width(100%).height(100%).justifyContent(FlexAlign.Cent…

时序深入之CPR(Clock Pessimism Removal)详解

目录 一、CPR概念 二、CPR的计算 三、CPR的开启关闭 四、CPR为0 ​五、参考资料 一、CPR概念 在时序报告的目标时钟路径中&#xff0c;会有一行数据clock pesssimism&#xff0c;第一次见可能都会对这个概念感到疑惑 同样在每条时序路径的summary中&#xff0c;clock pat…

吴恩达机器学习笔记:第 7 周-12支持向量机(Support Vector Machines)12.4-12.6

目录 第 7 周 12、 支持向量机(Support Vector Machines)12.4 核函数 112.5 核函数 212.6 使用支持向量机 第 7 周 12、 支持向量机(Support Vector Machines) 12.4 核函数 1 回顾我们之前讨论过可以使用高级数的多项式模型来解决无法用直线进行分隔的分类 问题&#xff1a; …

CentOS7升级openssl

文章目录 一 系统环境二 操作步骤三 版本检查 一 系统环境 公司服务器等保要求&#xff0c;修复openssl的高危漏洞。 本机使用centos7.9系统&#xff0c;openssl版本是1.0.2k&#xff0c;计划升级到1.1.1q 在执行下列操作前&#xff0c;务必要打快照做好备份&#xff0c;以防升…

java使用fasttext实现文本分类

1.fastText 官网 fastText是一个用于有效学习单词表示和句子分类的库fastText建立在现代Mac OS和Linux发行版上。因为它使用了c 11的特性&#xff0c;所以它需要一个具有良好的c11支持的编译器 2.创建maven项目 maven配置: <?xml version"1.0" encoding&quo…

14 Php学习:表单

表单 PHP 表单是用于收集用户输入的工具&#xff0c;通常用于网站开发。PHP 可以与 HTML 表单一起使用&#xff0c;用于处理用户提交的数据。通过 PHP 表单&#xff0c;您可以创建各种类型的表单&#xff0c;包括文本输入框、复选框、下拉菜单等&#xff0c;以便用户可以填写和…

AIGC算法1:Layer normalization

1. Layer Normalization μ E ( X ) ← 1 H ∑ i 1 n x i σ ← Var ⁡ ( x ) 1 H ∑ i 1 H ( x i − μ ) 2 ϵ y x − E ( x ) Var ⁡ ( X ) ϵ ⋅ γ β \begin{gathered}\muE(X) \leftarrow \frac{1}{H} \sum_{i1}^n x_i \\ \sigma \leftarrow \operatorname{Var}(…

Java | Leetcode Java题解之第35题搜索插入位置

题目&#xff1a; 题解&#xff1a; class Solution {public int searchInsert(int[] nums, int target) {int n nums.length;int left 0, right n - 1, ans n;while (left < right) {int mid ((right - left) >> 1) left;if (target < nums[mid]) {ans mi…