首页 > 其他分享 >【设计模式】创建型设计模式之工厂模式(简单工厂、工厂方法、抽象工厂、go简单实例)

【设计模式】创建型设计模式之工厂模式(简单工厂、工厂方法、抽象工厂、go简单实例)

时间:2024-06-12 21:30:21浏览次数:9  
标签:逻辑 对象 创建 代码 模式 工厂 go 设计模式

一般情况下,工厂模式分为三种更为细分的类型:简单工厂、工厂方法和抽象工厂。其中,前两者的方法原理比较简单,在实际的项目里也比较常用;而抽象工厂的原理稍微复杂,在实际的项目中相对也不常用。所以,我们今天重点是前两种工厂模式,

简单工厂

在下面这段代码里,我们根据配置文件的后缀(json,xml,yaml, properties),选择不同的解析器(JsonRuleConfigParser, XmlRuleConfigParser),将存在文件中的配置解析成内存对象RuleConfig

在这里插入图片描述

为了让代码逻辑更加清晰、可读性更好,我们要善于把功能独立的代码块封装成函数,按照这个设计思路,我们可以把代码中设计parse创建的部分逻辑剥离出来,抽象成createParse()函数。重构之后的代码如下所示:

在这里插入图片描述

为了让类的职责更加单一、代码更加清晰,我们还可以进一步将createParse()函数剥离到一个独立的类中,让这个类只负责对象的创建。而这个类就是简单工厂模式类。

在这里插入图片描述

大部分工厂类都是以Factory这个单词结尾的,但是也不是必须的;工厂类中创建对象的方法一般是create开头,比如代码里的createParser,但是也有的命名为getInstance(),createInstance(),newInstance,我们根据具体的场景和习惯来命名就好。

在上面的代码里,每次调用RuleConfigParserFactory的createParse的时候,都需要创建一个新的parser。实际上,如果parser可以复用,为了节省内存和对象创建的时间,我们可以把parser事先创建好缓存起来,类似单例模式和工厂模式的结合。

在这里插入图片描述

思考1:对于上面两种简单工厂模式的实现方法,如果我们要添加新的parser,那势必要改动到RuleConfigParserFactory的代码,那是不是违反开闭原则呢?

如果不是需要频繁地添加新的parser,只是偶尔修改RuleConfigParserFactory代码,稍微不符合开闭原则,也是可以接受的。

思考2:第一种实现里,有一组if分支判断逻辑,是否需要用多态或其他设计模式替代?

如果if分支不是很多,代码中有if分支也是可以接受的。

使用多态或设计模式替代if分支判断逻辑,虽然提高了代码的扩展性,但是也增加了类的个数,牺牲了代码的可读性。

工厂方法

思考3:如果必须将if分支逻辑去掉,那应该怎么办?

比较经典的方法就是利用多态 重构后的代码如下:

在这里插入图片描述

这就是工厂方法模式的典型代码实现,当新增一种parser的时候,只需要新增一个实现了IRuleConfigParserFactory接口的Factory类即可。所以,工厂方法模式比简单工厂更符合开闭原则

但是实际上上述的方法存在很大的问题,在使用这些工厂类的时候,如何写load函数呢?

在这里插入图片描述

工厂类对象的创建逻辑又耦合进了load函数里,跟最初的代码版本很相似。那么怎么解决这个问题呢?

可以为工厂类再创建一个简单工厂,也就是工厂的工厂,用来创建工厂类对象。代码如下,其中RuleConfigParserFactoryMap类是创建工厂对象的工厂类,RuleConfigParserFactoryMap返回的是缓存好的单例工厂对象。

在这里插入图片描述

在这里插入图片描述

当我们需要添加新的规则配置解析器时,我们只需要创建新的parser类和parser factory类,并且在RuleConfigParserFactoryMap类里,将新的parser factory对象添加到cachedFactories里即可,代码的改动很少,基本上符合开闭原则。

实际上,对于规则配置文件解析这个应用场景来说,工厂模式需要额外创建诸多Factory类,也会增加代码的复杂性。而且,每个Factory类只是做简单的new操作,功能单薄,也没必要设计成独立的类。所以在这个应用场景下,简单工厂模式简单好用,比工厂方法模式更加合适。

思考:什么时候该用工厂方法模式,而非简单工厂模式?

对象的创建逻辑比较复杂,不是简单的new一下就可以,而是要组装其他类对象,做各种初始化操作的时候,推荐使用工厂方法模式,将复杂的创建逻辑拆分到多个工厂类中,让每个工厂类都不至于过于复杂。

某些场景下,如果对象不可复用,工厂类每次都要返回不同的对象,如果用简单工厂模式实现,就只能选择第一种包含if分支逻辑的实现方式。如果我们还想避免烦人的if-else分支逻辑,就推荐使用工厂方法模式

抽象工厂

在简单工厂和工厂方法中,类只有一种分类方法。比如在规则配置解析的例子里,解析器类只会根据配置文件格式(Json,xml,yaml)来分类。但是,如果类又两种分类方式。比如,我们既可以按照配置文件格式来分类,也可以按照解析的对象(Rule规则配置还是System系统配置)来分类,就会对应下面8个parser类。

在这里插入图片描述

针对这种特殊的场景,如果还是继续用工厂方法来实现的话,我们要针对每个parser都编写一个工厂类,也就是要编写8个工厂类。如果未来还要增加针对业务配置的解析器,比如IBizConfigParser,还需要再增加4个工厂类。

抽象工厂就是针对这种非常特殊的场景而诞生的,我们可以让一个工厂负责创建多个不同类型的对象(IRuleConfigParser、ISystemConfigParser 等),而不是只创建一种parser对象。
在这里插入图片描述
在这里插入图片描述

总结

当创建逻辑比较复杂,是一个大工程的时候,考虑使用工厂模式,封装对象的创建过程,将对象的创建和使用分离。

什么情况是创建逻辑比较复杂呢?

  1. 类似规则配置解析的例子,代码里存在if-else判断,动态的根据不同的类型创建不同的对象。

  2. 单个对象本身的创建过程比较复杂,比如要组合其他类对象,做各种初始化操作。

对于情况1,当每个对象的创建逻辑都比较简单,推荐使用简单工厂模式,将多个对象的创建逻辑放到一个工厂类里,当每个对象的创建逻辑都比较复杂的时候,为了避免设计一个庞大的简单工厂类,推荐使用工厂方法模式,将创建逻辑拆分的更细,每个对象的创建逻辑独立到各自的工厂类里。

对于情况2, 因为单个对象本身的创建逻辑比较复杂,推荐使用工厂方法模式。

对于其他情况,如果创建对象的逻辑并不复杂,直接通过new来创建对象就可以了,不需要使用工厂模式。

现在,我们上升一个思维层面来看工厂模式,它的作用无外乎下面这四个。这也是判断要不要使用工厂模式的最本质的参考标准。

封装变化: 创建逻辑有可能变化,封装成工厂类之后,创建逻辑的变更对调用者透明。

代码复用: 创建代码抽离到独立的工厂类之后可以复用。

隔离复杂性: 封装复杂的创建逻辑,调用者无需了解如何创建对象。

控制复杂度: 将创建代码抽离出来,让原本的函数或类职责更单一,代码更简洁。

go代码示例

package _2_factory

import "fmt"

// 奶茶店
type MilkTeaShop interface {
	GetMilkTea()
}

type HoneySnowIceCity struct{}

func (h *HoneySnowIceCity) GetMilkTea() {
	fmt.Println("蜜雪冰城的柠檬水")
}

type ChaPanda struct{}

func (c *ChaPanda) GetMilkTea() {
	fmt.Println("茶百道的芒芒生打椰")
}

func NewMilkTeaShop(name string) MilkTeaShop {
	switch name {
	case "mixue":
		return &HoneySnowIceCity{}
	case "cha":
		return &ChaPanda{}
	}
	return nil
}


func Test_SimpleFactory(t *testing.T) {
	NewMilkTeaShop("mixue").GetMilkTea()
	NewMilkTeaShop("cha").GetMilkTea()
}
//=== RUN   Test_SimpleFactory
//蜜雪冰城的柠檬水
//茶百道的芒芒生打椰

抽象工厂

package _2_factory

import "fmt"

type Lunch interface {
	Cook()
}

type rise struct {
}

func (c *rise) Cook() {
	fmt.Println("it is a rise.")
}

type Tomato struct {
}

func (c *Tomato) Cook() {
	fmt.Println("it is Tomato.")
}

type LunchFactory interface {
	CreateFood() Lunch
	CreateVegetable() Lunch
}

type simpleLunchFactory struct {
}

func NewSimpleShapeFactory() LunchFactory {
	return &simpleLunchFactory{}
}

func (s *simpleLunchFactory) CreateFood() Lunch {
	return &rise{}
}

func (s *simpleLunchFactory) CreateVegetable() Lunch {
	return &Tomato{}
}

func TestNewSimpleShapeFactory(t *testing.T) {
	factory := NewSimpleShapeFactory()
	food := factory.CreateFood()
	food.Cook()

	vegetable := factory.CreateVegetable()
	vegetable.Cook()
}

//it is a rise.
//it is Tomato.

参考:
设计模式之美
go设计模式

标签:逻辑,对象,创建,代码,模式,工厂,go,设计模式
From: https://blog.csdn.net/LightOfNight/article/details/139636977

相关文章

  • gophish钓鱼
    目录环境介绍安装1、设置发件人的邮箱2、编辑钓鱼邮件模板3、制作钓鱼页面4、设置目标用户5、创建钓鱼事件6、查看结果参考1参考2参考3Gophish官网并下载适用于Linux的版本环境介绍Ubuntu22图形化/16G/4U/120G172.16.186.137/24安装rambo@test1:~$mkdirgophish&&cd......
  • golang sync.Map 与使用普通的 map 的区别
     使用sync.Map与普通的Gomap主要有以下几点区别:1.并发安全性普通map:在没有外部同步的情况下,不是并发安全的。在多goroutine访问时,如果没有适当的锁或其他同步机制保护,可能会导致数据竞争和未定义行为。sync.Map:是并发安全的。它内部实现了必要的同步机制,允许多......
  • 1. Two Sum Go实现
    在数组中找到2个数之和等于给定值的数字,结果返回2个数字在数组中的下标。1.解法1时间复杂度O(n^2)直接两次遍历所有节点,进行求和比较代码如下:functwoSum(nums[]int,targetint)[]int{ res:=make([]int,2,2) fori:=0;i<len(nums);i++{ forj:=i+1;j<len......
  • 基于GO语言,K8s+gRPC实战云原生微服务
    介绍K8s在云原生微服务开发中,作为微服务治理框架越来越受企业的青睐,掌握该技术解决方案更有竞争力,课程从企业实际开发中提取精髓,从K8s、gRPC底层原理剖析到服务治理解决方案设计落地,到云上部署,更平滑的学习曲线,助力你成为云原生开发领域的牛人。你将学到掌握整套K8s微服务......
  • 深入理解java设计模式之单例模式
    这里写目录标题概述单例模式是什么单例模式的使用场景单例模式的优缺点单例模式的几种实现方式饿汉式懒汉式双重检查锁定机制静态内部类枚举使用容器几种可能破坏单例类的方法多线程环境下的竞争条件使用反射机制使用序列化多个类加载器......
  • Go - pflag and viper
    pflagisadrop-inreplacementofGo'snativeflagpackage.Ifyouimportpflagunderthename"flag"thenallcodeshouldcontinuetofunctionwithnochanges.importflag"github.com/spf13/pflag"Thereisoneexceptiontothis:......
  • Dragon Boat Festival
    BeforetheDragonBoatFestival,mygrandmamadedozensofzongzimadeupofreedleaves,polishedglutinousriceandmeat.Mygrandmamakesdeliciouszongzieveryyear.Idon’tknowhowtomakeit.TheDragonBoatFestivalcamesilently.Onthemorning,......
  • go http请求
    funcSend(notificationmodel.Notification,defaultRobotstring)(errerror){markdown,robotURL,err:=transformer.TransformToMarkdown(notification)iferr!=nil{return}data,err:=json.Marshal(markdown)iferr!=nil{......
  • 边缘网关在智能制造工厂中的创新应用及效果-天拓四方
    在数字化浪潮席卷之下,智能制造工厂正面临着前所未有的数据挑战与机遇。边缘网关,作为数据处理与传输的关键节点,在提升工厂运营效率、确保数据安全方面发挥着日益重要的作用。本文将通过一个具体案例,详细阐述边缘网关在智能制造工厂中的创新应用及其带来的显著成效。案例背景某......
  • Go版RuoYi
     RuoYi-Go  https://github.com/Kun-GitHub/RuoYi-Go1.关于我个人介绍2.介绍后端用Go写的RuoYi权限管理系统(功能正在持续实现)后端 Gitee地址3.前端RuoYi-Vue3官方前端Vue3版4.Go后端技术栈(持续在对齐项目,在补充)功能框架是否采用备注配置......