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

news2025/7/18 11:38:50

文章目录

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

抽象工厂方法模式

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

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

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

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

产品族与产品等级结构

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

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

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

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

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

抽象工厂方法模式的实现

package main

import "fmt"

// ---------- 抽象层 ----------
type AbstractApple interface {
	ShowApple()
}

type AbstractBanana interface {
	ShowBanana()
}

type AbstractPear interface {
	ShowPear()
}

// 抽象工厂: 定义了工厂可以生产哪些抽象产品
type AbstractFactory interface {
	CreateApple() AbstractApple
	CreateBanana() AbstractBanana
	CreatePear() 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 AbstractFactory
	liaoningFac = &LiaoningFactory{}

	var liaoningApple AbstractApple
	liaoningApple = liaoningFac.CreateApple()
	liaoningApple.ShowApple()

	var liaoningBanana AbstractBanana
	liaoningBanana = liaoningFac.CreateBanana()
	liaoningBanana.ShowBanana()

	var liaoningPear AbstractPear
	liaoningPear = 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.coloradmin.cn/o/2387514.html

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

相关文章

一台手机怎样实现多IP上网?方法有多种

在数字时代,多IP上网已成为许多手机用户的刚需。本文将详细介绍如何通过不同技术手段实现手机多IP上网,帮助读者根据实际需求选择适合的解决方案。 一、为什么一台手机要实现多IP上网 手机实现多IP上网的典型场景包括: ①防止同一IP操作多个…

Linux 527 重定向 2>1 rsync定时同步(未完)

rsync定时同步 配环境 关闭防火墙、selinux systemctl stop firewalld systemctl disable firewalld setenforce0 vim /etc/SELINUX/config SELINUXdisable515 设置主机名 systemctl set-hostname code systemctl set-hostname backup 配静态ip rsync 需要稳定的路由表和端…

3DVR拍摄指南:从理论到实践

3DVR拍摄指南:从理论到实践 3D虚拟现实(Virtual Reality,简称VR)作为近年来迅速崛起的高新技术,通过电脑模拟产生一个三维空间的虚拟世界,为使用者提供视觉、听觉乃至触觉的全方位感官模拟,使用户仿佛身临…

leetcode hot100刷题日记——21.不同路径

和20题一样的思路link 题解&#xff1a; class Solution { public:int dfs(int i,int j,vector<vector<int>>&memo){//超过了边界&#xff0c;return 0if(i<0||j<0){return 0;}//从&#xff08;0&#xff0c;0&#xff09;到&#xff08;0&#xff0c;0…

通过JS模板引擎实现动态模块组件(Vite+JS+Handlebars)

1. 引言 在上一篇文章《实现一个前端动态模块组件(Vite原生JS)》中&#xff0c;笔者通过原生的JavaScript实现了一个动态的模块组件。但是这个实现并不完善&#xff0c;最大的问题就是功能逻辑并没有完全分开。比如模块的HTML&#xff1a; <div class"category-secti…

fvm install 下载超时 过慢 fvm常用命令、flutter常用命令

Git 配置问题 确保 Git 使用的是 HTTPS&#xff0c;而不是 SSH。如果你有 .gitconfig&#xff0c;确保没有配置奇怪的代理&#xff1a; git config --global --get http.proxy git config --global --get https.proxy如果有代理设置且不需要&#xff0c;取消代理&#xff1a;…

Python正则表达式:30秒精通文本处理

一、概述 1. 含义 正则表达式是一种记录文本规则的代码工具&#xff0c;用于描述字符串的结构和模式。它广泛应用于字符串的匹配、查找、替换、提取等操作。 2. 特点 语法复杂&#xff1a;符号多、规则灵活&#xff0c;可读性较差。功能强大&#xff1a;可以精确控制字符串…

Introduction to SQL

目录 SQL特点 ​编辑 Select-From-Where Statements Meaning of Single-Relation Query Operational Semantics * In SELECT clauses Complex Conditions in WHERE Clause PATTERNS NULL Values Three-Valued Logic Multirelation Queries Aggregations NULL’s Ig…

计算机视觉---YOLOv3

YOLOv3讲解 一、YOLOv3 核心架构与创新 YOLOv3&#xff08;2018年发布&#xff09;在YOLOv2基础上进行了全面升级&#xff0c;通过多尺度预测、更强大的骨干网络和优化的分类损失函数&#xff0c;显著提升了检测精度&#xff0c;尤其是小目标检测能力&#xff0c;同时保持了实…

#RabbitMQ# 消息队列进阶

目录 消息可靠性 一 生产者的可靠性 1 生产者的重连 2 生产者的确认 (1 Confirm* (2 Return 二 MQ的可靠性 1 数据持久化 2 Lazy Queue* 三 消费者的可靠性 1 消费者确认机制 2 消费失败处理 3 业务幂等性 四 延迟消息 消息可靠性 在消息队列中&#xff0c;可靠性…

【深度学习】损失“三位一体”——从 Fisher 的最大似然到 Shannon 的交叉熵再到 KL 散度,并走进 PET·P-Tuning微调·知识蒸馏的实战

一页速览&#xff1a; 1912 Fisher 用最大似然把「让数据出现概率最高」变成参数学习&#xff1b; 1948 Shannon 把交叉熵解释成「最短平均编码长度」&#xff1b; 1951 Kullback-Leibler 用相对熵量化「多余信息」。 三条历史线落到今天深度学习同一个损失——交叉熵。 也…

5 分钟速通密码学!

让我们开始第一部分&#xff1a;密码学基础 (Cryptography Basics)。 第一部分&#xff1a;密码学基础 (Cryptography Basics) 1. 什么是密码学&#xff1f; 想象一下&#xff0c;在古代战争中&#xff0c;将军需要向远方的部队传递作战指令。如果直接派人送信&#xff0c;信…

Linux——IP协议

1. 现实意义 • IP协议&#xff1a;提供一种能力&#xff0c;把数据报从主机A跨网络送到主机B • TCP/IP协议&#xff1a;核心功能&#xff0c;把数据100%可靠的从主机A跨网络送到主机B 注&#xff1a;TCP协议负责百分百可靠&#xff0c;通过三次握手、滑动窗口、拥塞控制、延…

Linux信号量(32)

文章目录 前言一、POSIX 信号量信号量的基础知识信号量的基本操作 二、基于环形队列实现生产者消费者模型环形队列单生产单消费模型多生产多消费模型 总结 前言 加油&#xff0c;加油&#xff01;&#xff01;&#xff01; 一、POSIX 信号量 信号量的基础知识 互斥、同步 不只…

技术视界 | 打造“有脑有身”的机器人:ABC大脑架构深度解析(上)

ABC大脑架构&#xff1a;连接大模型与物理世界的具身智能新范式 在具身智能和类人机器人技术快速发展的背景下&#xff0c;如何高效整合“大模型的认知理解能力”与“对真实物理世界的精准控制”&#xff0c;成为当前智能体系统设计中最具挑战性也是最关键的问题之一。尽管大语…

使用堡塔和XShell

使用堡塔和XShell 一、SSH协议介绍 SSH为SecureShell的缩写&#xff0c;由IETF的网络小组(NetworkWorkingGroup)所制定;SSH为建立在应用层基础上的安全协议。SSH是较可靠&#xff0c;专为远程登录会话和其他网络服务提供安全性的协议。利用SSH协议可以有效防止远程管理过程中…

软件项目交付阶段,验收报告记录了什么?有哪些标准要求?

软件项目交付阶段&#xff0c;验收报告扮演着至关重要的角色&#xff0c;它相当于一份详尽的“成绩单”&#xff0c;具体记录了项目完成的具体情况以及是否达到了既定的标准。 项目基本信息 该环节将展示软件项目的核心信息&#xff0c;包括项目名称、开发团队构成、项目实施…

LightGBM的python实现及参数优化

文章目录 1. LightGBM模型参数介绍2. 核心优势3. python实现LightGBM3.1 基础实现3.1.1 Scikit-learn接口示例3.1.2 Python API示例 3.2 模型调优3.2.1 GridSearchCV简介3.2.2 LightGBM超参调优3.2.3 GridSearchCV寻优结果解读 在之前的文章 Boosting算法【AdaBoost、GBDT 、X…

封装渐变堆叠柱状图组件附完整代码

组件功能 这是一个渐变堆叠柱状图组件&#xff0c;主要功能包括&#xff1a; 在一根柱子上同时显示高、中、低三种危险级别数据使用渐变色区分不同危险级别&#xff08;高危红色、中危橙色、低危蓝色&#xff09;悬停显示详细数据信息&#xff08;包括总量和各级别数据&#…

山东大学软件学院创新项目实训开发日志——第十三周

目录 1.开展prompt工程&#xff0c;创建个性化AI助理&#xff0c;能够基于身份实现不同角度和语言风格的回答。 2.对输出进行格式化&#xff0c;生成特定格式的会议计划文档。 3.学习到的新知识 本阶段我所做的工作 1.开展prompt工程&#xff0c;创建个性化AI助理&#xff…