【设计模式】创建型模式——简单工厂模式

文章目录

  • 一、创建型模式
    • 1. 包含模式
      • 1.1 工厂模式
      • 1.2 建造者模式
      • 1.3 原型模式
      • 1.4 单例模式
  • 二、工厂模式
    • 1. 概要
      • 1.1 意图
      • 1.2 主要解决问题
      • 1.3 何时使用
      • 1.4 如何解决
      • 1.5 关键代码
      • 1.6 使用场景
      • 1.7 优点
      • 1.8 缺点
      • 1.9 简单工厂实现
    • 2. 简单工厂模式
      • 2.1 模式结构
      • 2.2 实例
      • 2.3 模式定义
      • 2.4 模式分析
      • 2.5 适用情况
      • 2.6 特点
        • 2.6.1 优点
        • 2.6.2 缺点


一、创建型模式

创建型模式对类的实例化过程进行了抽象,能够将软件模块中对象的创建和对象的使用分离。

为了使软件的结构更加清晰,外界对于这些对象只需要知道它们共同的结构,而不清楚其具体的实现细节,使整个系统的设计更加符合单一职责原则。

创建型模型在创建什么(What),由谁创建(Who),何时创建(When)等方面都为软件设计者提供了尽可能大的灵活性。

创建型模式隐藏了类的实例的创建细节,通过隐藏对象如何被创建和组合在一起达到使整个系统独立的目的。

1. 包含模式

1.1 工厂模式

  1. 简单工厂模式
  2. 工厂方法模式
  3. 抽象工厂模式

1.2 建造者模式

1.3 原型模式

1.4 单例模式

二、工厂模式

工厂模式是最常用的设计模式之一,提供了一种创建对象的方式,使得对象的创建过程与对象的使用过程分离。
工厂模式提供了一种创建对象的方式,而无需指定要创建的具体类。
通过使用工厂模式,可以将对象的创建逻辑封装在一个工厂类中,而不是在客户端代码中直接实例化对象,这样可以提高代码的可维护性和可扩展性。

1. 概要

概要内容偏书面化,读起来有点绕,看看代码实现就很容易理解啦~~我个人体验概要不需要读,直接看代码也能明白做法的意图啦

1.1 意图

定义一个创建对象的接口,让其子类决定实例化哪一个具体的类。工厂模式使对象的创建过程延迟到子类。

1.2 主要解决问题

接口选择的问题。

1.3 何时使用

当需要在不同条件下创建不同实例时。

1.4 如何解决

通过让子类实现工厂接口,返回一个抽象的产品。

1.5 关键代码

对象的创建过程在子类中实现。

1.6 使用场景

  1. 日志记录:日志可能记录到本地硬盘、系统事件、远程服务器等,用户可以选择记录日志的位置。
  2. 数据库访问:当用户不知道最终系统使用哪种数据库,或者数据库可能变化时。
  3. 连接服务器的框架设计:需要支持 “POP3”、“IMAP”、“HTTP” 三种协议,可以将这三种协议作为产品类,共同实现一个接口。

1.7 优点

  1. 调用者只需要知道对象的名称即可创建对象。
  2. 扩展性高,如果需要增加新产品,只需扩展一个工厂类即可。
  3. 屏蔽了产品的具体实现,调用者只关心产品的接口。

1.8 缺点

每次增加一个产品时,都需要增加一个具体类和对应的工厂,使系统中类的数量成倍增加,增加了系统的复杂度和具体类的依赖。

1.9 简单工厂实现

创建一个 Shape 接口和实现 Shape 接口的实体类。下一步是定义工厂类 ShapeFactory。

FactoryPatternDemo 类使用 ShapeFactory 来获取 Shape 对象。它将向 ShapeFactory 传递信息(CIRCLE / RECTANGLE / SQUARE),以便获取它所需对象的类型。
在这里插入图片描述

package mainimport "fmt"// Shape 接口
type Shape interface {Draw()
}// Rectangle 结构体
type Rectangle struct{}// Draw 方法实现
func (r *Rectangle) Draw() {fmt.Println("Inside Rectangle::draw() method.")
}// Square 结构体
type Square struct{}// Draw 方法实现
func (s *Square) Draw() {fmt.Println("Inside Square::draw() method.")
}// Circle 结构体
type Circle struct{}// Draw 方法实现
func (c *Circle) Draw() {fmt.Println("Inside Circle::draw() method.")
}// ShapeFactory 结构体
type ShapeFactory struct{}// GetShape 方法
func (f *ShapeFactory) GetShape(shapeType string) Shape {switch shapeType {case "CIRCLE":return &Circle{}case "RECTANGLE":return &Rectangle{}case "SQUARE":return &Square{}default:return nil}
}// 主函数
func main() {shapeFactory := &ShapeFactory{}// 获取 Circle 的对象,并调用它的 Draw 方法shape1 := shapeFactory.GetShape("CIRCLE")shape1.Draw()// 获取 Rectangle 的对象,并调用它的 Draw 方法shape2 := shapeFactory.GetShape("RECTANGLE")shape2.Draw()// 获取 Square 的对象,并调用它的 Draw 方法shape3 := shapeFactory.GetShape("SQUARE")shape3.Draw()
}

2. 简单工厂模式

2.1 模式结构

简单工厂模式包含如下角色:

1. Factory:工厂角色
工厂角色负责实现创建所有实例的内部逻辑。

2. Product:抽象产品角色
抽象产品角色是所创建的所有对象的父类,负责描述所有实例所共有的公共接口。

3. ConcreteProduct:具体产品角色
具体产品角色是创建目标,所有创建的对象都充当这个角色的某个具体类的实例。
在这里插入图片描述

2.2 实例

工厂类负责创建对象,客户端只需调用工厂方法。

package mainimport "fmt"// Shape 接口
type Shape interface {Draw()
}// Circle 结构体
type Circle struct{}func (c *Circle) Draw() {fmt.Println("Drawing a Circle")
}// Rectangle 结构体
type Rectangle struct{}func (r *Rectangle) Draw() {fmt.Println("Drawing a Rectangle")
}// ShapeFactory 简单工厂
type ShapeFactory struct{}func (f *ShapeFactory) GetShape(shapeType string) Shape {switch shapeType {case "CIRCLE":return &Circle{}case "RECTANGLE":return &Rectangle{}default:return nil}
}// 主函数
func main() {factory := &ShapeFactory{}shape1 := factory.GetShape("CIRCLE")shape1.Draw()shape2 := factory.GetShape("RECTANGLE")shape2.Draw()
}

2.3 模式定义

简单工厂模式(Simple Factory Pattern):又称为静态工厂方法(Static Factory Method)模式,它属于类创建型模式。
在简单工厂模式中,可以根据参数的不同返回不同类的实例。
简单工厂模式专门定义一个类来负责创建其他类的实例,被创建的实例通常都具有共同的父类。

2.4 模式分析

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

2.5 适用情况

  • 工厂类负责创建的对象比较少:由于创建的对象较少,不会造成工厂方法中的业务逻辑太过复杂。
  • 客户端只知道传入工厂类的参数,对于如何创建对象不关心:客户端既不需要关心创建细节,甚至连类名都不需要记住,只需要知道类型所对应的参数。

2.6 特点

2.6.1 优点
  • 工厂类含有必要的判断逻辑,可以决定在什么时候创建哪一个产品类的实例,客户端可以免除直接创建产品对象的责任,而仅仅“消费”产品;简单工厂模式通过这种做法实现了对责任的分割,它提供了专门的工厂类用于创建对象。
  • 客户端无须知道所创建的具体产品类的类名,只需要知道具体产品类所对应的参数即可,对于一些复杂的类名,通过简单工厂模式可以减少使用者的记忆量。
  • 通过引入配置文件,可以在不修改任何客户端代码的情况下更换和增加新的具体产品类,在一定程度上提高了系统的灵活性。
2.6.2 缺点
  • 由于工厂类集中了所有产品创建逻辑,一旦不能正常工作,整个系统都要受到影响。
  • 使用简单工厂模式将会增加系统中类的个数,在一定程序上增加了系统的复杂度和理解难度。
  • 系统扩展困难,一旦添加新产品就不得不修改工厂逻辑,在产品类型较多时,有可能造成工厂逻辑过于复杂,不利于系统的扩展和维护。
  • 简单工厂模式由于使用了静态工厂方法,造成工厂角色无法形成基于继承的等级结构。

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

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

相关文章

python实战一:合并多个Excel中数据

看不懂可以跟着视频学一下,关于基础课程项目也可以私聊我! 视频源码: 链接:https://pan.quark.cn/s/2055653f735b 之前我们已经学习了如何读取和写入Execl数据,今天我们来用一个实例来进行表格的合并。如下是 2021年…

Durid解析SQL语句

在外面的需求中,有很多需要解析SQL语句的地方,我们采用Durid来进行解析。 Durid可以将sql进行详细的拆分成多个部分 解析where解析SQLSelectItem解析update语句解析limit解析group by 还可以动态修改sql,比如在原sql上增加条件修改sql运行的…

YOLOv9改进策略【模型轻量化】| ShufflenetV2,通过通道划分构建高效网络

一、本文介绍 本文记录的是基于ShufflenetV2的YOLOv9目标检测轻量化改进方法研究。FLOPs是评价模型复杂独的重要指标,但其无法考虑到模型的内存访问成本和并行度,因此本文在YOLOv9的基础上引入ShufflenetV2,使其在在保持准确性的同时提高模型…

多线程篇(基本认识 - 锁优化)(持续更新迭代)

目录 一、前言 二、阿里开发手册 三、synchronized 锁优化的背景 四、Synchronized的性能变化 1. Java5之前:用户态和内核态之间的切换 2. java6开始:优化Synchronized 五、锁升级 1. 无锁 2. 偏向锁 2.1. 前言 2.2. 什么是偏向锁 2.3. 偏向…

ChatTTS 长音频合成和本地部署2种方式,让你的“儿童绘本”发声的实战教程(文末有福利)

接上文(GLM-4-Flash 大模型 API 免费了,手把手构建“儿童绘本”应用实战(附源码)),老牛同学通过 GLM-4-Flash 文生文和 CogView-3 文生图大模型,和大家一起编写了一个图文并茂的儿童绘本应用&am…

先进制造aps专题二十五 openai的ai大模型设计也使用了aps用的并行遗传算法

openai的ai大模型设计是不是很神秘? 其实不神秘,看国内最新的一个团队招人 新型研究机构,AI团队招收ai大模型算子搜索、大模型训练与推理算法 ai大模型算子搜索:方向:算子辅助搜索工具的实现与开发、群体遗传算法研…

从默默无名到销量激增,极越CEO哭了

文 | 智能相对论 作者 | 周晚 8月28日,极越汽车CEO夏一平在接受媒体采访的时候泪洒当场! 倒不是因为极越汽车销量寥寥,而是因为突然销量激增,他带领的极越团队走出低谷,当他们回过头去看今年上半年的巨大压力&#…

欧拉 函数

互质: 互质是公约数只有1的两个整数,叫做互质整数。公约数只有1的两个自然数,叫做互质自然数,后者是前者特殊情况。 (1和-1与所有整数互质,而且它们是唯一与0互质的整数) 互质的判断方法&…

面试(九)

目录 一. 僵尸进程/孤儿进程/守护进程 二 线程的同步和异步 三. 线程间通信 3.1 共享内存 3.2 互斥锁 3.3 条件变量 3.4 信号量 3.5 读写锁 3.6 事件 3.7 线程局部存储 四. 进程间通信 3.1 管道 3.2 消息队列 3.3. 共享内存 3.4 信号量 3.5 套接字 3.6 内存映…

redis的RDB快照配置详解

Redis 的 RDB(Redis Database Backup)快照是一种数据持久化方式,它可以在指定时间间隔内将内存中的数据快照保存到磁盘。这样,即使Redis服务器意外宕机或重启,数据也能从最近的快照中恢复。RDB快照的配置主要涉及以下几…

如何快速判断Excel中选区跨页?

实例需求:应用开发过程中,需要校验选中区域,要求选中区域不能跨页,即选中区域分布在两个不同的页面中。 示例代码如下。 Sub Demo()Dim rSelect As Range, oHP As HPageBreakDim UpCell As Range, DownCell As RangeSet rSelect…

1500+ HuggingFace API token暴露,使数百万 Meta-Llama、Bloom 等大模型用户面临风险

HugginingFace 和 GitHub 是AI开发人员的重要平台,因此我们决定对这些平台做更为深入的研究。在这项研究中,我们发现了数以千计的 API 令牌,这些令牌暴露给恶意行为者,使数百万最终用户可能受到攻击。 本文的主要内容包括&#x…

【Leetcode 2341 】 数组能形成多少数对 —— 去重

给你一个下标从 0 开始的整数数组 nums 。在一步操作中,你可以执行以下步骤: 从 nums 选出 两个 相等的 整数从 nums 中移除这两个整数,形成一个 数对 请你在 nums 上多次执行此操作直到无法继续执行。 返回一个下标从 0 开始、长度为 2 的…

MySQL高阶练习题1- 寻找面试候选人

目录 题目 准备数据 分析数据 实现代码 总结 题目 返回 所有面试候选人 的姓名 name 和邮件 mail 。当用户满足以下两个要求中的 任意一条 ,其成为 面试候选人 : 该用户在 连续三场及更多 比赛中赢得 任意 奖牌。该用户在 三场及更多不同的 比赛中赢得 金牌&…

python实战三-提取Word数据到Excel

视频源码: 链接:https://pan.quark.cn/s/83db5bb15383 一个文件夹下有大量会议通知文件,为word文件,文件格式都是一致的,现在要将文件中的一些字段提取出来汇总到Excel文件中。 会议通知文件格式如下: 要提…

C++(1)基础语法

C(1)之基础语法 Author: Once Day Date: 2024年8月29日 一位热衷于Linux学习和开发的菜鸟,试图谱写一场冒险之旅,也许终点只是一场白日梦… 漫漫长路,有人对你微笑过嘛… 全系列文章可参考专栏: 源码分析_Once-Day的博客-CSDN博客 参考文…

java中哪些集合可以使用 Iterator 进行遍历

在 JAVA 集合中&#xff0c;分为两种顶层接口&#xff0c;Collection 和 Map。 而 Iterator 是针对 Collection 的。 因此&#xff0c;JAVA 中所有 Collection 的子类都可以使用 Iterator。 像 List、Set、Queue、Deque 都可以使用。 使用例子&#xff1a; List<Integer…

通用后台管理系统实战演示(Vue3 + element-plus)汇总篇一

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…

【Windows学习笔记】1:OneCore和Windows API

1 OneCore 主流Windows跑在PC上&#xff0c;Xbox跑的是从Windows 2000分支出来的版本&#xff0c;Windows Phone跑的是基于Windows CE&#xff08;实时操作系统&#xff09;的版本。 为了维护和扩展这些code方便&#xff0c;微软将kernels和base platform集成在一起叫OneCore…

55-java多线程future

java多线程future 在Java中&#xff0c;Future 接口和 Callable 函数式接口一起被用来支持异步计算。Future 表示一个可能完成的异步计算的结果&#xff0c;通过 get 方法可以获取计算的结果。 下面是一个使用 Future 的简单示例&#xff1a; import java.util.concurrent.*…