C# 工厂模式学习
工厂模式(Factory Pattern)是一种创建型设计模式,它提供了一种创建对象的接口,而不是通过具体类来实例化对象。工厂模式可以将对象的创建过程封装起来,使代码更具有灵活性和可扩展性。
工厂模式有几种常见的实现方式:
-
简单工厂模式(Simple Factory Pattern): 简单工厂模式通过一个工厂类来决定创建哪种具体类的实例。这个工厂类通常提供一个静态方法,根据传入的参数创建相应的对象。
-
工厂方法模式(Factory Method Pattern): 工厂方法模式定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。工厂方法使一个类的实例化延迟到其子类。
-
抽象工厂模式(Abstract Factory Pattern): 抽象工厂模式提供一个接口,用于创建相关或依赖对象的家族,而无需明确指定具体类。通过使用抽象工厂模式,一个类可以实例化一组相关对象,而不需要知道它们的具体类。
简单工厂模式示例
假设我们有一个动物园项目,需要创建不同的动物对象:
// 动物接口
public interface IAnimal
{void Speak();
}// 具体的动物类
public class Dog : IAnimal
{public void Speak(){Console.WriteLine("Woof!");}
}public class Cat : IAnimal
{public void Speak(){Console.WriteLine("Meow!");}
}// 简单工厂类
public static class AnimalFactory
{public static IAnimal CreateAnimal(string animalType){switch (animalType.ToLower()){case "dog":return new Dog();case "cat":return new Cat();default:throw new ArgumentException("Unknown animal type");}}
}// 使用示例
class Program
{static void Main(string[] args){IAnimal animal = AnimalFactory.CreateAnimal("dog");animal.Speak(); // 输出:Woof!}
}
工厂方法模式示例
假设我们有一个动物园项目,不同的子类需要创建不同的动物对象:
// 动物接口
public interface IAnimal
{void Speak();
}// 具体的动物类
public class Dog : IAnimal
{public void Speak(){Console.WriteLine("Woof!");}
}public class Cat : IAnimal
{public void Speak(){Console.WriteLine("Meow!");}
}// 工厂接口
public interface IAnimalFactory
{IAnimal CreateAnimal();
}// 具体工厂类
public class DogFactory : IAnimalFactory
{public IAnimal CreateAnimal(){return new Dog();}
}public class CatFactory : IAnimalFactory
{public IAnimal CreateAnimal(){return new Cat();}
}// 使用示例
class Program
{static void Main(string[] args){IAnimalFactory factory = new DogFactory();IAnimal animal = factory.CreateAnimal();animal.Speak(); // 输出:Woof!}
}
抽象工厂模式示例
假设我们有一个动物园项目,需要创建一组相关的对象(例如,动物及其食物):
// 动物接口
public interface IAnimal
{void Speak();
}// 具体的动物类
public class Dog : IAnimal
{public void Speak(){Console.WriteLine("Woof!");}
}public class Cat : IAnimal
{public void Speak(){Console.WriteLine("Meow!");}
}// 食物接口
public interface IFood
{void Get();
}// 具体的食物类
public class DogFood : IFood
{public void Get(){Console.WriteLine("Dog food");}
}public class CatFood : IFood
{public void Get(){Console.WriteLine("Cat food");}
}// 抽象工厂接口
public interface IAnimalFactory
{IAnimal CreateAnimal();IFood CreateFood();
}// 具体工厂类
public class DogFactory : IAnimalFactory
{public IAnimal CreateAnimal(){return new Dog();}public IFood CreateFood(){return new DogFood();}
}public class CatFactory : IAnimalFactory
{public IAnimal CreateAnimal(){return new Cat();}public IFood CreateFood(){return new CatFood();}
}// 使用示例
class Program
{static void Main(string[] args){IAnimalFactory factory = new DogFactory();IAnimal animal = factory.CreateAnimal();IFood food = factory.CreateFood();animal.Speak(); // 输出:Woof!food.Get(); // 输出:Dog food}
}
以上是三种工厂模式的基本示例,可以根据具体需求选择合适的工厂模式来实现代码的创建和管理。如果希望在增加新动物类型时尽量减少对现有类的修改,推荐使用工厂方法模式。工厂方法模式的设计使得每新增一种动物,只需增加一个对应的工厂类和具体的动物类,而无需修改已有的代码,从而符合开闭原则(即对扩展开放,对修改关闭)。
使用工厂方法模式
下面是一个更完善的工厂方法模式示例,展示了如何在增加新动物时,尽量减少对现有代码的修改。
// 动物接口
public interface IAnimal
{void Speak();
}// 具体的动物类
public class Dog : IAnimal
{public void Speak(){Console.WriteLine("Woof!");}
}public class Cat : IAnimal
{public void Speak(){Console.WriteLine("Meow!");}
}// 新增的动物类
public class Bird : IAnimal
{public void Speak(){Console.WriteLine("Tweet!");}
}// 工厂接口
public interface IAnimalFactory
{IAnimal CreateAnimal();
}// 具体工厂类
public class DogFactory : IAnimalFactory
{public IAnimal CreateAnimal(){return new Dog();}
}public class CatFactory : IAnimalFactory
{public IAnimal CreateAnimal(){return new Cat();}
}// 新增的动物工厂类
public class BirdFactory : IAnimalFactory
{public IAnimal CreateAnimal(){return new Bird();}
}// 使用示例
class Program
{static void Main(string[] args){List<IAnimalFactory> factories = new List<IAnimalFactory>{new DogFactory(),new CatFactory(),new BirdFactory() // 新增的工厂只需在这里添加};foreach (var factory in factories){IAnimal animal = factory.CreateAnimal();animal.Speak();}}
}
在这个示例中,新增一种动物只需:
- 创建新的具体动物类,例如
Bird。 - 创建对应的工厂类,例如
BirdFactory。 - 在使用的地方添加新的工厂实例,例如在
factories列表中添加new BirdFactory()。
这样做的好处是每增加一个新动物类型,不需要修改现有的工厂类或具体的动物类,只需要添加新的类和工厂即可,从而降低了代码修改的风险和复杂度。
使用反射和配置来进一步减少修改
如果希望在增加动物时连代码都不需要改动,可以考虑使用反射和配置文件的方式。通过配置文件定义动物类型和对应的工厂类,然后使用反射动态加载:
// 动物接口和具体的动物类(同上)// 工厂接口和具体工厂类(同上)// 使用反射加载工厂类
class Program
{static void Main(string[] args){// 假设配置文件中定义了动物类型和对应的工厂类var factoryTypes = new List<string>{"DogFactory","CatFactory","BirdFactory" // 配置文件中新增的工厂类};var factories = new List<IAnimalFactory>();foreach (var factoryType in factoryTypes){var type = Type.GetType(factoryType);if (type != null && typeof(IAnimalFactory).IsAssignableFrom(type)){var factory = (IAnimalFactory)Activator.CreateInstance(type);factories.Add(factory);}}foreach (var factory in factories){IAnimal animal = factory.CreateAnimal();animal.Speak();}}
}
接口与继承结合使用
工厂模式主要使用了接口、继承,在C#中,接口和继承是面向对象编程的重要概念。接口定义了一组方法和属性,而继承允许一个类从另一个类继承其成员。接口可以实现多重继承,而类只能继承一个基类。通常情况下,接口和继承可以结合使用,以充分利用它们各自的优点。通过这种方式,基类可以提供一些通用的实现,而接口可以定义特定的行为。
// 接口
public interface IAnimal
{void Speak();void Eat();
}// 基类
public class Animal
{public void Sleep(){Console.WriteLine("Sleeping...");}
}// 派生类实现接口
public class Dog : Animal, IAnimal
{public void Speak(){Console.WriteLine("Woof!");}public void Eat(){Console.WriteLine("Dog is eating.");}
}public class Cat : Animal, IAnimal
{public void Speak(){Console.WriteLine("Meow!");}public void Eat(){Console.WriteLine("Cat is eating.");}
}// 使用示例
class Program
{static void Main(string[] args){IAnimal dog = new Dog();dog.Speak(); // 输出:Woof!dog.Eat(); // 输出:Dog is eating.IAnimal cat = new Cat();cat.Speak(); // 输出:Meow!cat.Eat(); // 输出:Cat is eating.// 使用基类方法Animal animalDog = (Animal)dog;animalDog.Sleep(); // 输出:Sleeping...Animal animalCat = (Animal)cat;animalCat.Sleep(); // 输出:Sleeping...}
}
总结
- 接口:定义了一组必须实现的方法和属性,没有实现代码。支持多重继承,使得类可以实现多个接口。
- 继承:用于从现有类创建新类,继承基类的成员。每个类只能有一个基类,但可以实现多个接口。
- 结合使用:通过将接口和继承结合使用,可以实现代码的高复用性和灵活性。
通过上述示例,可以看到如何使用接口和继承来设计灵活且可扩展的应用程序结构。这样既能充分利用基类的通用功能,又能通过接口实现特定的行为。
相关文章:
C# 工厂模式学习
工厂模式(Factory Pattern)是一种创建型设计模式,它提供了一种创建对象的接口,而不是通过具体类来实例化对象。工厂模式可以将对象的创建过程封装起来,使代码更具有灵活性和可扩展性。 工厂模式有几种常见的实现方式&…...
AI生成微信职业头像
加油,新时代打工人! 真别说,还挺好看的 https://chatglm.cn/main/alltoolsdetail...
遥感图像的深度学习的任务类型
在遥感图像的深度学习任务中,利用深度学习技术处理和分析遥感图像已经成为一个重要的研究方向。遥感图像来自卫星、无人机等设备,包含了丰富的地球表面信息。以下是遥感图像深度学习中的主要任务类型: 1. 图像分类(Image Classif…...
162.二叉树:填充每个节点的下一个右侧节点指针(力扣)
代码解决 /* // Definition for a Node. class Node { public:int val;Node* left;Node* right;Node* next;Node() : val(0), left(NULL), right(NULL), next(NULL) {}Node(int _val) : val(_val), left(NULL), right(NULL), next(NULL) {}Node(int _val, Node* _left, Node* _…...
NLP(20)--知识图谱+实体抽取
前言 仅记录学习过程,有问题欢迎讨论 基于LLM的垂直领域问答方案: 特点:不是通用语料;准确度要求高,召回率可以低(转人工);拓展性和可控性(改变特定内容的回答…...
【mysql数据库】mycat中间件
MyCat 简介 Mycat 是数据库 中间件 。 1、 数据库中间件 中间件 是一类连接软件组件和应用的计算机软件, 以便于软件各部件之间的沟通 。 例子 Tomcat web 中间件 。 数据库 中间件 连接 java 应用程序和数据库 2、 为什么要用 Mycat ① Java 与数据库紧耦合 …...
满帮集团 Eureka 和 ZooKeeper 的上云实践
作者:胡安祥 满帮集团,作为“互联网物流”的平台型企业,一端承接托运人运货需求,另一端对接货车司机,提升货运物流效率。2021 年美股上市,成为数字货运平台上市第一股。根据公司年报,2021 年&a…...
ubuntu中彻底删除mysql (配置文件删除可选)
ubuntu中彻底删除mysql (配置文件删除可选) 对于此类即搜即用的分享文章,也不过多赘述,直接依次按照下面的操作执行即可: 一、删除 mysql 数据文件 sudo rm /var/lib/mysql/ -R二、删除 mysql 配置文件 sudo rm /etc/mysql/ -R三、查看 m…...
根据模板和git commit自动生成日·周·月·季报
GitHub - qiaotaizi/dailyreport: 日报生成器 GitHub - yurencloud/daily: 程序员专用的日报、周报、月报、季报自动生成器! config.json: { "Author": "gitname", "Exclude": ["update:", "add:", "…...
matlab GUI界面设计
【实验内容】 用MATLAB的GUI程序设计一个具备图像边缘检测功能的用户界面,该设计程序有以下基本功能: (1)图像的读取和保存。 (2)设计图形用户界面,让用户对图像进行彩色图像到灰度图像的转换…...
MyBatis 面试题
一、什么是 Mybatis? 1、Mybatis 是一个半 ORM(对象关系映射)框架,它内部封装了 JDBC,开发时 只需要关注 SQL 语句本身,不需要花费精力去处理加载驱动、创建连接、创建 statement 等繁杂的过程。程序员直接编写原生态 sql,可以严格控制 sql 执行性 能,灵活度高。 …...
C#根据数据量自动排版标签的样例
这是一个C#根据数据量自动排版标签的样例 using System; using System.Collections.Generic; using System.Data.SqlClient; using System.Drawing; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; using HslCommuni…...
【网络安全】Web安全基础 - 第一节:使用软件及环境介绍
VMware VMware,是全球云基础架构和移动商务解决方案的佼佼者。 VMware可是一个总部位于美国加州帕洛阿尔托的计算机虚拟化软件研发与销售企业呢。简单来说,它就是通过提供虚拟化解决方案,让企业在数据中心改造和公有云整合业务上更加得心应…...
Mac下载docker
先安装homebrew Mac下载Homebrew-CSDN博客 然后输入以下命令安装docker brew install --cask --appdir/Applications docker 期间需要输入密码。输入完等待即可...
k8s_设置dns
配置k8s dns 在 Kubernetes 集群中,CoreDNS 是默认的 DNS 服务器,它负责处理集群内所有的 DNS 请求。 kubectl edit cm coredns -n kube-system (此命令修改coredns 配置) kubectl describe cm coredns -n kube-system(此命令查看coredns 配…...
翻译《The Old New Thing》- What a drag: Dragging a virtual file (HGLOBAL edition)
What a drag: Dragging a virtual file (HGLOBAL edition) - The Old New Thing (microsoft.com)https://devblogs.microsoft.com/oldnewthing/20080318-00/?p23083 Raymond Chen 2008年03月18日 拖拽虚拟文件(HGLOBAL 版本) 现在我们已经对简单的数据…...
SA316系列音频传输模块-传输距离升级音质不打折
SA316是思为无线研发的一款远距离音频传输模块,音频采样率为48K,传输距离可达200M。为了满足更多用户需求,思为无线在SA316基础上进一步增加传输距离推出SA316F30。相比SA316性能,同样其采用48K采样,-96dBm灵敏度&…...
【机器学习】智能选择的艺术:决策树在机器学习中的深度剖析
在机器学习的分类和回归问题中,决策树是一种广泛使用的算法。决策树模型因其直观性、易于理解和实现,以及处理分类和数值特征的能力而备受欢迎。本文将解释决策树算法的概念、原理、应用、优化方法以及未来的发展方向。 🚀时空传送门 &#x…...
电脑缺少运行库,无法启动程序
在我们使用一些软件的时候,由于电脑缺少一些运行库,导致无法启动应用软件,此时需要我们安装缺少的运行库。 比如当电脑提示: Cannot load library Qt5Xlsx.dll 我们就需要下载C得运行库,以满足软件运行需要。 下载链…...
【计算机软考_初级篇】每日十题2
各位老师大家好,软考对于日常的知识储备和企业中的考试,或者说在校大学生来说,那用处是非常大的!!那么下面我们进入正题,软考呢是分两种语言,java和C,对于其他语言目前还没ÿ…...
conda相比python好处
Conda 作为 Python 的环境和包管理工具,相比原生 Python 生态(如 pip 虚拟环境)有许多独特优势,尤其在多项目管理、依赖处理和跨平台兼容性等方面表现更优。以下是 Conda 的核心好处: 一、一站式环境管理:…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
Vue3 + Element Plus + TypeScript中el-transfer穿梭框组件使用详解及示例
使用详解 Element Plus 的 el-transfer 组件是一个强大的穿梭框组件,常用于在两个集合之间进行数据转移,如权限分配、数据选择等场景。下面我将详细介绍其用法并提供一个完整示例。 核心特性与用法 基本属性 v-model:绑定右侧列表的值&…...
ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放
简介 前面两期文章我们介绍了I2S的读取和写入,一个是通过INMP441麦克风模块采集音频,一个是通过PCM5102A模块播放音频,那如果我们将两者结合起来,将麦克风采集到的音频通过PCM5102A播放,是不是就可以做一个扩音器了呢…...
Swagger和OpenApi的前世今生
Swagger与OpenAPI的关系演进是API标准化进程中的重要篇章,二者共同塑造了现代RESTful API的开发范式。 本期就扒一扒其技术演进的关键节点与核心逻辑: 🔄 一、起源与初创期:Swagger的诞生(2010-2014) 核心…...
C++使用 new 来创建动态数组
问题: 不能使用变量定义数组大小 原因: 这是因为数组在内存中是连续存储的,编译器需要在编译阶段就确定数组的大小,以便正确地分配内存空间。如果允许使用变量来定义数组的大小,那么编译器就无法在编译时确定数组的大…...
《C++ 模板》
目录 函数模板 类模板 非类型模板参数 模板特化 函数模板特化 类模板的特化 模板,就像一个模具,里面可以将不同类型的材料做成一个形状,其分为函数模板和类模板。 函数模板 函数模板可以简化函数重载的代码。格式:templa…...
使用LangGraph和LangSmith构建多智能体人工智能系统
现在,通过组合几个较小的子智能体来创建一个强大的人工智能智能体正成为一种趋势。但这也带来了一些挑战,比如减少幻觉、管理对话流程、在测试期间留意智能体的工作方式、允许人工介入以及评估其性能。你需要进行大量的反复试验。 在这篇博客〔原作者&a…...
Ubuntu系统复制(U盘-电脑硬盘)
所需环境 电脑自带硬盘:1块 (1T) U盘1:Ubuntu系统引导盘(用于“U盘2”复制到“电脑自带硬盘”) U盘2:Ubuntu系统盘(1T,用于被复制) !!!建议“电脑…...
CppCon 2015 学习:Time Programming Fundamentals
Civil Time 公历时间 特点: 共 6 个字段: Year(年)Month(月)Day(日)Hour(小时)Minute(分钟)Second(秒) 表示…...
