c# 设计模式--抽象工厂模式 (Abstract Factory)

定义

抽象工厂模式是一种创建型设计模式,它提供了一种创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。抽象工厂模式强调的是对象族的创建,而不是单一对象的创建。

用例写法

假设我们有一个场景,需要根据不同的平台(如 Windows 和 MacOS)创建不同的用户界面组件(如按钮和文本框)。

// 用户界面组件接口
public interface IButton
{void Render();
}public interface ITextBox
{void Render();
}// Windows用户界面组件
public class WindowsButton : IButton
{public void Render() => Console.WriteLine("Rendering Windows Button...");
}public class WindowsTextBox : ITextBox
{public void Render() => Console.WriteLine("Rendering Windows TextBox...");
}// MacOS用户界面组件
public class MacOSButton : IButton
{public void Render() => Console.WriteLine("Rendering MacOS Button...");
}public class MacOSTextBox : ITextBox
{public void Render() => Console.WriteLine("Rendering MacOS TextBox...");
}// 抽象工厂接口
public interface IUIComponentFactory
{IButton CreateButton();ITextBox CreateTextBox();
}// Windows用户界面组件工厂
public class WindowsUIComponentFactory : IUIComponentFactory
{public IButton CreateButton() => new WindowsButton();public ITextBox CreateTextBox() => new WindowsTextBox();
}// MacOS用户界面组件工厂
public class MacOSUIComponentFactory : IUIComponentFactory
{public IButton CreateButton() => new MacOSButton();public ITextBox CreateTextBox() => new MacOSTextBox();
}// 配置类
public class Configuration
{public string OS { get; set; }
}// 客户端代码
class Program
{static void Main(string[] args){Configuration config = new Configuration { OS = "Windows" };IUIComponentFactory factory;if (config.OS == "Windows"){factory = new WindowsUIComponentFactory();}else if (config.OS == "MacOS"){factory = new MacOSUIComponentFactory();}else{throw new ArgumentException("Unsupported operating system");}IButton button = factory.CreateButton();ITextBox textBox = factory.CreateTextBox();button.Render();textBox.Render();}
}

类图:

IButton
+void Render()
ITextBox
+void Render()
WindowsButton
+void Render()
WindowsTextBox
+void Render()
MacOSButton
+void Render()
MacOSTextBox
+void Render()
IUIComponentFactory
+IButton CreateButton()
+ITextBox CreateTextBox()
WindowsUIComponentFactory
+IButton CreateButton()
+ITextBox CreateTextBox()
MacOSUIComponentFactory
+IButton CreateButton()
+ITextBox CreateTextBox()
Configuration
-string OS
+Configuration()
Program
+static void Main(string[] args)

解释

  1. 用户界面组件接口 (IButtonITextBox):定义了渲染按钮和文本框的方法。
  2. 具体用户界面组件 (WindowsButtonWindowsTextBoxMacOSButtonMacOSTextBox):实现了具体的按钮和文本框渲染逻辑。
  3. 抽象工厂接口 (IUIComponentFactory):定义了创建按钮和文本框的方法。
  4. 具体工厂类 (WindowsUIComponentFactoryMacOSUIComponentFactory):实现了具体的按钮和文本框创建逻辑。
  5. 配置类 (Configuration):用于存储操作系统的配置信息。
  6. 客户端代码 (Program):根据配置信息选择合适的工厂类,并使用工厂类创建具体的用户界面组件,然后调用其渲染方法。

用途

  1. 创建一系列相关对象:当需要创建一系列相关或相互依赖的对象时,使用抽象工厂模式可以确保这些对象的一致性。
  2. 解耦对象的创建和使用:客户端代码只需要依赖抽象工厂接口,不需要知道具体的产品类,从而降低了耦合度。
  3. 支持多平台:适用于需要在不同平台上创建不同版本的对象的场景。

优点

  1. 易于交换产品族:可以在运行时动态地切换不同的产品族,只需更换工厂对象即可。
  2. 支持开闭原则:增加新的产品族时,只需要添加新的具体工厂类,不需要修改现有的代码,符合开闭原则。
  3. 高度模块化:每个产品族的创建逻辑都封装在对应的工厂类中,模块化程度高。

缺点

  1. 系统复杂度增加:引入了更多的接口和类,系统结构变得更复杂。
  2. 增加代码量:每增加一个新的产品族,都需要增加相应的工厂类和产品类。
  3. 产品族扩展困难:如果需要增加新的产品类型,必须修改所有的工厂类,这违反了开闭原则。

适用场景

  1. 产品族多:当系统中有多个产品族,且每个产品族中的产品需要一起使用时。
  2. 多平台支持:需要在不同平台上创建不同版本的对象时。
  3. 对象创建逻辑复杂:对象的创建逻辑较为复杂,需要集中管理和控制时。

实际开发中的应用

1. 多平台用户界面框架

假设我们正在开发一个多平台用户界面框架,需要根据不同的操作系统创建不同的用户界面组件。

// 用户界面组件接口
public interface IButton
{void Render();
}public interface ITextBox
{void Render();
}// Windows用户界面组件
public class WindowsButton : IButton
{public void Render() => Console.WriteLine("Rendering Windows Button...");
}public class WindowsTextBox : ITextBox
{public void Render() => Console.WriteLine("Rendering Windows TextBox...");
}// MacOS用户界面组件
public class MacOSButton : IButton
{public void Render() => Console.WriteLine("Rendering MacOS Button...");
}public class MacOSTextBox : ITextBox
{public void Render() => Console.WriteLine("Rendering MacOS TextBox...");
}// 抽象工厂接口
public interface IUIComponentFactory
{IButton CreateButton();ITextBox CreateTextBox();
}// Windows用户界面组件工厂
public class WindowsUIComponentFactory : IUIComponentFactory
{public IButton CreateButton() => new WindowsButton();public ITextBox CreateTextBox() => new WindowsTextBox();
}// MacOS用户界面组件工厂
public class MacOSUIComponentFactory : IUIComponentFactory
{public IButton CreateButton() => new MacOSButton();public ITextBox CreateTextBox() => new MacOSTextBox();
}// 配置类
public class Configuration
{public string OS { get; set; }
}// 客户端代码
class Program
{static void Main(string[] args){Configuration config = new Configuration { OS = "Windows" };IUIComponentFactory factory;if (config.OS == "Windows"){factory = new WindowsUIComponentFactory();}else if (config.OS == "MacOS"){factory = new MacOSUIComponentFactory();}else{throw new ArgumentException("Unsupported operating system");}IButton button = factory.CreateButton();ITextBox textBox = factory.CreateTextBox();button.Render();textBox.Render();}
}

解释

  1. 用户界面组件接口 (IButtonITextBox):定义了渲染按钮和文本框的方法。
  2. 具体用户界面组件 (WindowsButton、WindowsTextBox、MacOSButton、MacOSTextBox):实现了具体的按钮和文本框渲染逻辑。
  3. 抽象工厂接口 (IUIComponentFactory):定义了创建按钮和文本框的方法。
  4. 具体工厂类 (WindowsUIComponentFactoryMacOSUIComponentFactory):实现了具体的按钮和文本框创建逻辑。
  5. 配置类 (Configuration):用于存储操作系统的配置信息。
  6. 客户端代码 (Program):根据配置信息选择合适的工厂类,并使用工厂类创建具体的用户界面组件,然后调用其渲染方法。

2. 数据库操作框架

假设我们正在开发一个数据库操作框架,需要根据不同的数据库类型创建不同的数据访问对象。

// 数据访问接口
public interface IDbConnection
{void Connect();void Disconnect();
}public interface IDbCommand
{void Execute();
}// MySQL数据库操作
public class MySqlConnection : IDbConnection
{public void Connect() => Console.WriteLine("Connecting to MySQL database...");public void Disconnect() => Console.WriteLine("Disconnecting from MySQL database...");
}public class MySqlCommand : IDbCommand
{public void Execute() => Console.WriteLine("Executing MySQL command...");
}// SQL Server数据库操作
public class SqlServerConnection : IDbConnection
{public void Connect() => Console.WriteLine("Connecting to SQL Server database...");public void Disconnect() => Console.WriteLine("Disconnecting from SQL Server database...");
}public class SqlCommand : IDbCommand
{public void Execute() => Console.WriteLine("Executing SQL Server command...");
}// 抽象工厂接口
public interface IDbFactory
{IDbConnection CreateConnection();IDbCommand CreateCommand();
}// MySQL数据库工厂
public class MySqlDbFactory : IDbFactory
{public IDbConnection CreateConnection() => new MySqlConnection();public IDbCommand CreateCommand() => new MySqlCommand();
}// SQL Server数据库工厂
public class SqlServerDbFactory : IDbFactory
{public IDbConnection CreateConnection() => new SqlServerConnection();public IDbCommand CreateCommand() => new SqlCommand();
}// 配置类
public class Configuration
{public string DatabaseType { get; set; }
}// 客户端代码
class Program
{static void Main(string[] args){Configuration config = new Configuration { DatabaseType = "MySQL" };IDbFactory factory;if (config.DatabaseType == "MySQL"){factory = new MySqlDbFactory();}else if (config.DatabaseType == "SQLServer"){factory = new SqlServerDbFactory();}else{throw new ArgumentException("Unsupported database type");}IDbConnection connection = factory.CreateConnection();IDbCommand command = factory.CreateCommand();connection.Connect();command.Execute();connection.Disconnect();}
}

总结

抽象工厂模式通过提供一个创建一系列相关或相互依赖对象的接口,使得对象的创建更加灵活和一致。虽然它增加了系统的复杂度,但在需要创建多个相关对象的场景中,抽象工厂模式能够显著提高代码的可维护性和扩展性。

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

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

相关文章

Java线程的interrupt中断、wait-notify/all(源码级分析)

实例方法: interrupt()方法是设置结束阻塞(sleep、wait等),并且设置中断标记true isInterrupted()判断当前是否中断 静态方法: Thread.interrupted():调用这个方法的线程中断标记位还原为false 那么好,既然上面的方法作用是清…

Burp Suite 实战指南:Proxy 捕获与修改流量、HTTP History 筛选与分析

声明! 学习视频来自B站up主 **泷羽sec** 有兴趣的师傅可以关注一下,如涉及侵权马上删除文章,笔记只是方便各位师傅的学习和探讨,文章所提到的网站以及内容,只做学习交流,其他均与本人以及泷羽sec团队无关&a…

12月第1周AI资讯

阅读时间:3-4min 更新时间:2024.12.2-2024.12.6 目录 OpenAI CEO Sam Altman 预告“12天OpenAI”系列活动 腾讯HunyuanVideo:130亿参数的开源视频生成模型 李飞飞的World Labs发布空间智能技术预览版 中科院联手腾讯打造“AI带货王”AnchorCrafter OpenAI CEO Sam Alt…

从零开始学TiDB(1) 核心组件架构概述

首先TiDB深度兼容MySQL 5.7 1. TiDB Server SQL语句的解析与编译:首先一条SQL语句最先到达的地方是TiDB Server集群,TiDB Server是无状态的,不存储数据,SQL 发过来之后TiDB Server 负责 解析,优化,编译 这…

记录一次使用git无权限的问题排查

正常的配置了公私钥之后,在gitlab中也存储了配对的公钥,但当使用git clone 时,总是报无权限 由于在这台机器中添加了多个公私钥,有点复杂,我们可以使用命令 ssh -vvvT 调试一下 ssh -vvvT yourGitlabAddr

python调用GPT-4o实时音频 Azure OpenAI GPT-4o Audio and /realtime

发现这块网上信息很少,记录一下 微软azure入口 https://learn.microsoft.com/zh-cn/azure/ai-services/openai/realtime-audio-quickstart?pivotsprogramming-language-ai-studio sdk文档 https://github.com/azure-samples/aoai-realtime-audio-sdk?tabread…

fastadmin 后台插件制作方法

目录 一:开发流程 二:开发过程 (一):后台功能开发 (二):功能打包到插件目录 (三):打包插件 (四):安装插件…

Kafka单机及集群部署及基础命令

目录 一、 Kafka介绍1、kafka定义2、传统消息队列应用场景3、kafka特点和优势4、kafka角色介绍5、分区和副本的优势6、kafka 写入消息的流程 二、Kafka单机部署1、基础环境2、iptables -L -n配置3、下载并解压kafka部署包至/usr/local/目录4、修改server.properties5、修改/etc…

Docker部署的gitlab升级的详细步骤(升级到17.6.1版本)

文章目录 一、Gitlab提示升级信息二、老版本的docker运行gitlab命令三、备份老版本Gitlab数据四、确定升级路线五、升级(共分3个版本升级)5.1 升级第一步(17.1.2 > 17.3.7)5.2 升级第二步(17.3.7 > 17.5.3)5.3 升级第三步(17.5.3 > 17.6.1) 六、web端访问gitlab服务 一…

在Java的xml的sql语句里面的某一个参数是list集合的时候

经常在Java里面,遇到这样的问题,sql的一个查询语句,它的某一个参数是一个List集合,然而,在xml.mapper文件里面的时候,不知道如何去组成这个查询语句,不知道兄弟们是否经常忘记如何去写这个语句&…

前端技术(23) : 聊天页面

来源: GPT生成之后微调 效果图 HTML代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>聊天</t…

内存图及其画法

所有的文件都存在硬盘上&#xff0c;首次使用的时候才会进入内存 进程&#xff1a;有自己的Main方法&#xff0c;并且依赖自己Main运行起来的程序。独占一块内存区域&#xff0c;互不干扰。内存中有一个一个的进程。 操作系统只认识c语言。操作系统调度驱动管理硬件&#xff0…

树与图深度优先遍历——acwing

题目一&#xff1a;树的重心 846. 树的重心 - AcWing题库 分析 采用暴力枚举&#xff0c;试探每个点&#xff0c;除去之后&#xff0c;连通分量最大值是多少&#xff0c; 各个点的最大值找最小的 因为可以通过 dfs 来得到 根u以下点数&#xff0c;以及可以求各分树的点数&am…

Qt Qtablewidget 标题 QHeaderView 增加可选框 QcheckBox

创建自定义QHeaderView #pragma once#include <QObject> #include <QHeaderView> #include <QPainter> #include <QMouseEvent>class SSHeaderView : public QHeaderView {Q_OBJECTprivate:bool isChecked;int m_checkColIdx; public:SSHeaderView(i…

Data Uncertainty Learning in Face Recognition 论文阅读

Data Uncertainty Learning in Face Recognition 论文阅读 Abstract1. Introduction2. Related Work3. Methodology3.1. Preliminaries3.2. Classification-based DUL for FR3.3. Regression-based DUL for FR3.4. Discussion of Related Works 4. Experiments4.1. Datasets an…

基础算法——搜索与图论

搜索与图论 图的存储方式2、最短路问题2.1、Dijkstra算法&#xff08;朴素版&#xff09;2.2、Dijkstra算法&#xff08;堆优化版&#xff09;2.3、Bellman-Ford算法2.4、SPFA求最短路2.5、SPFA判负环2.6、Floyd算法 图的存储方式 2、最短路问题 最短路问题可以分为单源最短路…

典型常见的知识蒸馏方法总结一

来源&#xff1a;https://github.com/HobbitLong/RepDistiller收录的方法 NeurIPS2015: Distilling the Knowledge in a Neural Network 知识蒸馏开山之作&#xff0c;从logits中蒸馏知识&#xff0c;KL散度损失 ICLR2015&#xff1a;FitNets: Hints for Thin Deep Nets A h…

十、软件设计架构-微服务-服务调用Feign

文章目录 前言一、Feign介绍1. 什么是Feign2. 什么是Http客户端3. Feign 和 OpenFeign 的区别 二、Feign底层原理三、Feign工作原理详解1. 动态代理机制2. 动态代理的创建过程3. 创建详细流程4. FeignClient属性 四、Feign使用1. 常规调用2.日志打印3. 添加Header 前言 服务调…

在 LS-DYNA 中将应力转换为用户定义的坐标系

介绍 通常&#xff0c;使用 LS-DYNA 或 Ansys Mechanical 等仿真工具解决工程问题需要将张量结果与解析解进行比较。一个这样的例子是加压圆柱体&#xff0c;其中圆周应力或环状应力是感兴趣的主要应力度量。例如&#xff0c;如果对具有复杂端部处理的几何结构进行此类仿真&am…

树莓集团是如何链接政、产、企、校四个板块的?

树莓集团作为数字影像行业的积极探索者与推动者&#xff0c;我们通过多维度、深层次的战略举措&#xff0c;将政、产、企、校四个关键板块紧密链接在一起&#xff0c;实现了资源的高效整合与协同发展&#xff0c;共同为数字影像产业的繁荣贡献力量。 与政府的深度合作政府在产业…