当前位置: 首页 > article >正文

【创造型模式】抽象工厂方法模式

文章目录

  • 抽象工厂方法模式
    • 产品族与产品等级结构
    • 抽象工厂方法模式的角色和职责
    • 抽象工厂方法模式的实现
    • 抽象工厂方法模式的优缺点
    • 适用场景

抽象工厂方法模式

在这里插入图片描述
工厂方法模式引入了“工厂等级结构”,解决了简单工厂方法过分依赖单一工厂的问题。但是工厂方法模式存在的一个问题是,针对每一种产品,都需要设计一种工厂,比如对于“苹果”和“山东苹果”,我们就需要分别设计一个“苹果工厂”和“山东苹果工厂”。

工厂方法模式现有的问题在于:

  1. 但添加一个新产品的时候(比如葡萄),虽然遵循开闭原则,不修改现有的代码,但是需要添加大量的类(比如辽宁葡萄、河北葡萄、山东葡萄),而且需要添加相对应的工厂,增加了系统开销和维护成本。
  2. 如果同一个“地域”有多种产品,比如“辽宁葡萄”、“辽宁樱桃”、“辽宁苹果”,那么需要分别创建具体的工厂,而不能通过一个工厂来生产所有的产品。虽然该方法满足开闭原则,但是代码将会变得非常复杂,不够好。

抽象工厂方法模式的引入进一步优化了工厂方法模式。具体来说,抽象工厂方法模式引入了“产品等级结构”和“产品族”这两个概念。

产品族与产品等级结构

此处借用《Easy 搞定 Golang 设计模式》当中的原图:

  • 产品族:具有同一个地区、同一个厂商、同一个开发包、同一个组织模块等隶属关系,但是具备不同的特点或功能,这样的产品构成的集合,就是一个产品族。比如产自同一个产地的各种水果,就构成一个产品族。
  • 产品等级结构:具有相同特点或功能,但是隶属的组织模块不同,这样的产品结构,称为一个产品等级。不同产地的同一种水果就是同一个产品等级。

“抽象工厂方法模式”针对“产品族”进行产品生产。

抽象工厂方法模式的角色和职责

  • 抽象工厂角色:声明了一组用于创建一组产品的方法,每一个方法对应一种产品。
  • 具体工厂角色:实现了在抽象工厂中声明的创建产品的方法,生成一组具体的产品,这些产品构成一个产品族,每一个产品都位于某个产品结构当中。
  • 抽象产品角色:为每种产品声明接口,在抽象产品中声明了产品所具有的业务方法。
  • 具体产品角色:定义具体工厂生产的具体产品对象,实现抽象产品接口声明的业务方法。

抽象工厂方法模式的实现

package mainimport "fmt"// ---------- 抽象层 ----------
type AbstractApple interface {ShowApple()
}type AbstractBanana interface {ShowBanana()
}type AbstractPear interface {ShowPear()
}// 抽象工厂: 定义了工厂可以生产哪些抽象产品
type AbstractFactory interface {CreateApple() AbstractAppleCreateBanana() AbstractBananaCreatePear() AbstractPear
}// ---------- 实现层 ----------
type LiaoningApple struct{}func (la *LiaoningApple) ShowApple() {fmt.Println("Liaoning Apple")
}type LiaoningBanana struct{}func (lb *LiaoningBanana) ShowBanana() {fmt.Println("Liaoning Banana")
}type LiaoningPear struct{}func (lp *LiaoningPear) ShowPear() {fmt.Println("Liaoning Pear")
}type LiaoningFactory struct{}func (fa *LiaoningFactory) CreateApple() AbstractApple {return &LiaoningApple{}
}func (fa *LiaoningFactory) CreateBanana() AbstractBanana {return &LiaoningBanana{}
}func (fa *LiaoningFactory) CreatePear() AbstractPear {return &LiaoningPear{}
}type JiangsuApple struct{}func (ja *JiangsuApple) ShowApple() {fmt.Println("Jiangsu Apple")
}type JiangsuBanana struct{}func (jb *JiangsuBanana) ShowBanana() {fmt.Println("Jiangsu Banana")
}type JiangsuPear struct{}func (jp *JiangsuPear) ShowPear() {fmt.Println("Jiangsu Pear")
}type JiangsuFactory struct{}func (fa *JiangsuFactory) CreateApple() AbstractApple {return &JiangsuApple{}
}func (fa *JiangsuFactory) CreateBanana() AbstractBanana {return &JiangsuBanana{}
}func (fa *JiangsuFactory) CreatePear() AbstractPear {return &JiangsuPear{}
}type BeijingApple struct{}func (ba *BeijingApple) ShowApple() {fmt.Println("Beijing Apple")
}type BeijingBanana struct{}func (bb *BeijingBanana) ShowBanana() {fmt.Println("Beijing Banana")
}type BeijingPear struct{}func (pbp *BeijingPear) ShowPear() {fmt.Println("Beijing Pear")
}type BeijingFactory struct{}func (fa *BeijingFactory) CreateApple() AbstractApple {return &BeijingApple{}
}func (fa *BeijingFactory) CreateBanana() AbstractBanana {return &BeijingBanana{}
}func (fa *BeijingFactory) CreatePear() AbstractPear {return &BeijingPear{}
}func main() {// 创建辽宁的苹果, 香蕉, 梨等对象var liaoningFac AbstractFactoryliaoningFac = &LiaoningFactory{}var liaoningApple AbstractAppleliaoningApple = liaoningFac.CreateApple()liaoningApple.ShowApple()var liaoningBanana AbstractBananaliaoningBanana = liaoningFac.CreateBanana()liaoningBanana.ShowBanana()var liaoningPear AbstractPearliaoningPear = liaoningFac.CreatePear()liaoningPear.ShowPear()// 创建其他产区的水果jiangsuFac := &JiangsuFactory{}jiangsuApple := jiangsuFac.CreateApple()jiangsuApple.ShowApple()jiangsuBanana := jiangsuFac.CreateBanana()jiangsuBanana.ShowBanana()jiangsuPear := jiangsuFac.CreatePear()jiangsuPear.ShowPear()beijingFac := &BeijingFactory{}beijingApple := beijingFac.CreateApple()beijingApple.ShowApple()beijingBanana := beijingFac.CreateBanana()beijingBanana.ShowBanana()beijingPear := beijingFac.CreatePear()beijingPear.ShowPear()
}

抽象工厂方法模式的优缺点

优点:

  • 继承了工厂方法模式的优点;
  • 确保客户端始终只使用一个同产品族中的对象;
  • 增加新的产品族很方便,对于上例而言,可以直接设计一个新的工厂并实现抽象工厂的接口。

缺点:

  • 新增产品的等级结构复杂,需要对原有系统进行较大的修改,甚至可能修改抽象层的代码。

适用场景

  • 系统中有多于一个产品族,每次只使用某个产品族。
  • 产品等级结构问题,设计系统完成后,不会向系统中新增产品等级结构或删除已有的产品等级结构。
http://www.lryc.cn/news/2387514.html

相关文章:

  • 一台手机怎样实现多IP上网?方法有多种
  • 【FFmpeg+SDL】播放音频时,声音正常但是有杂音问题(已解决)
  • Linux 527 重定向 2>1 rsync定时同步(未完)
  • 3DVR拍摄指南:从理论到实践
  • OSI模型中的网络协议
  • 【C/C++】线程局部存储:原理与应用详解
  • 分块查找详解
  • leetcode hot100刷题日记——21.不同路径
  • Elasticsearch 如何实现跨数据中心的数据同步?
  • C语言学习笔记三 --- V
  • 通过JS模板引擎实现动态模块组件(Vite+JS+Handlebars)
  • 梯度消失和梯度爆炸的原因及解决办法
  • 欧拉定理:若 gcd(a,n)=1,则 a^φ(n)≡1(mod n)。
  • fvm install 下载超时 过慢 fvm常用命令、flutter常用命令
  • Python正则表达式:30秒精通文本处理
  • Introduction to SQL
  • 计算机视觉---YOLOv3
  • #RabbitMQ# 消息队列进阶
  • React从基础入门到高级实战:React 核心技术 - React Router:路由管理
  • 【深度学习】损失“三位一体”——从 Fisher 的最大似然到 Shannon 的交叉熵再到 KL 散度,并走进 PET·P-Tuning微调·知识蒸馏的实战
  • 5 分钟速通密码学!
  • Linux——IP协议
  • Lua 脚本在 Redis 中的运用-24 (使用 Lua 脚本实现原子计数器)
  • Linux信号量(32)
  • 技术视界 | 打造“有脑有身”的机器人:ABC大脑架构深度解析(上)
  • 使用堡塔和XShell
  • 软件项目交付阶段,验收报告记录了什么?有哪些标准要求?
  • LightGBM的python实现及参数优化
  • 封装渐变堆叠柱状图组件附完整代码
  • 分布式项目保证消息幂等性的常见策略