C#设计模式详解(2)——Factory Method(工厂方法)

news2025/1/22 16:15:28

文章目录

  • C#设计模式详解(2)——Factory Method(工厂方法)
    • 工厂方法模式
      • 1.1 概念
      • 1.2 意图
      • 1.3 问题
      • 1.4 解决方案
      • 1.5 工厂方法模式结构
      • 1.6 案例代码
      • 1.7 游戏开发中的应用

C#设计模式详解(2)——Factory Method(工厂方法)

工厂方法模式

1.1 概念

亦称: 虚拟构造函数、Virtual Constructor、Factory Method

1.2 意图

工厂方法模式是一种创建型设计模式, 其在父类中提供一个创建对象的方法, 允许子类决定实例化对象的类型。

1.3 问题

假设你正在开发一款物流管理应用。 最初版本只能处理卡车运输, 因此大部分代码都在位于名为 卡车的类中。

一段时间后, 这款应用变得极受欢迎。 你每天都能收到十几次来自海运公司的请求, 希望应用能够支持海上物流功能。

在程序中新增一个运输类会遇到问题

如果代码其余部分与现有类已经存在耦合关系, 那么向程序中添加新类其实并没有那么容易。

这可是个好消息。 但是代码问题该如何处理呢? 目前, 大部分代码都与 卡车类相关。 在程序中添加 轮船类需要修改全部代码。 更糟糕的是, 如果你以后需要在程序中支持另外一种运输方式, 很可能需要再次对这些代码进行大幅修改。

最后, 你将不得不编写繁复的代码, 根据不同的运输对象类, 在应用中进行不同的处理。

1.4 解决方案

工厂方法模式建议使用特殊的工厂方法代替对于对象构造函数的直接调用 (即使用 new运算符)。 不用担心, 对象仍将通过 new运算符创建, 只是该运算符改在工厂方法中调用罢了。 工厂方法返回的对象通常被称作 “产品”。

创建者类结构

子类可以修改工厂方法返回的对象类型。

乍看之下, 这种更改可能毫无意义: 我们只是改变了程序中调用构造函数的位置而已。 但是, 仔细想一下, 现在你可以在子类中重写工厂方法, 从而改变其创建产品的类型。

但有一点需要注意:仅当这些产品具有共同的基类或者接口时, 子类才能返回不同类型的产品, 同时基类中的工厂方法还应将其返回类型声明为这一共有接口。

产品对象层次结构

所有产品都必须使用同一接口。

举例来说, 卡车Truck和 轮船Ship类都必须实现 运输Transport接口, 该接口声明了一个名为 deliver交付的方法。 每个类都将以不同的方式实现该方法: 卡车走陆路交付货物, 轮船走海路交付货物。 陆路运输Road­Logistics类中的工厂方法返回卡车对象, 而 海路运输Sea­Logistics类则返回轮船对象。

使用工厂方法模式后的代码结构

只要产品类实现一个共同的接口, 你就可以将其对象传递给客户代码, 而无需提供额外数据。

调用工厂方法的代码 (通常被称为客户端代码) 无需了解不同子类返回实际对象之间的差别。 客户端将所有产品视为抽象的 运输 。 客户端知道所有运输对象都提供 交付方法, 但是并不关心其具体实现方式。

1.5 工厂方法模式结构

工厂方法模式结构

  1. 产品 (Product) 将会对接口进行声明。 对于所有由创建者及其子类构建的对象, 这些接口都是通用的。

  2. 具体产品 (Concrete Products) 是产品接口的不同实现。

  3. 创建者 (Creator) 类声明返回产品对象的工厂方法。 该方法的返回对象类型必须与产品接口相匹配。

    你可以将工厂方法声明为抽象方法, 强制要求每个子类以不同方式实现该方法。 或者, 你也可以在基础工厂方法中返回默认产品类型。

    注意, 尽管它的名字是创建者, 但它最主要的职责并不是创建产品。 一般来说, 创建者类包含一些与产品相关的核心业务逻辑。 工厂方法将这些逻辑处理从具体产品类中分离出来。 打个比方, 大型软件开发公司拥有程序员培训部门。 但是, 这些公司的主要工作还是编写代码, 而非生产程序员。

  4. 具体创建者 (Concrete Creators) 将会重写基础工厂方法, 使其返回不同类型的产品。

    注意, 并不一定每次调用工厂方法都会创建新的实例。 工厂方法也可以返回缓存、 对象池或其他来源的已有对象。

1.6 案例代码

(自己写的,有问题欢迎指出并且讨论)

类图

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xOrAH5dU-1669545517115)(C:\Users\JackiieWang\AppData\Roaming\Typora\typora-user-images\image-20221127183817280.png)]

IProduct

namespace _001_test_工厂模式
{
    public interface IProduct
    {
        public string Operation();
        public void RuneLine();
    }
}

ConcreteProduct1

using System;

namespace _001_test_工厂模式
{
    class ConcreteProduct1 : IProduct
    {
        public string Operation()
        {
            string result = "具体产品1的相关操作";
            return result;
        }

        public void RuneLine()
        {
            ProductOneThings();
        }

        public void ProductOneThings()
        {
            Console.WriteLine("这是专属于具体产品1独有的特性");
        }
    }
}

ConcreteProduct2

using System;

namespace _001_test_工厂模式
{
    class ConcreteProduct2 : IProduct
    {
        public string Operation()
        {
            var result = "具体产品2的相关操作";
            return result;
        }

        public void RuneLine()
        {
            ProductTwoThings();
            OnSale();
        }

        public void ProductTwoThings()
        {
            Console.WriteLine("这是专属于具体产品2独有的特性");
        }

        public void OnSale()
        {
            Console.WriteLine("这是具体产品2的销售方法");
        }
    }
}

ConCretor

namespace _001_test_工厂模式
{
    public abstract class ConCretor
    {
        public abstract IProduct FactoryMethod();

        public string SomeOperation()
        {
            var product = FactoryMethod();
            var result = "父类执行了ConCretorDoSomeThing()方法并调用了" + product.Operation();
            return result;
        }

        //...父类调用子类的操作流程
        public void RuneLine()
        {
           var product = FactoryMethod();
           product.RuneLine();
        }

    }
}

ConcreteCreator1

namespace _001_test_工厂模式
{
    class ConcreteCreator1 : ConCretor
    {
        public override IProduct FactoryMethod()
        {
            return new ConcreteProduct1();

            //...具体产品接口的变化可以在具体的产品内部去做操作
        }
    }
}

ConcreteCreator2

using System;
using System.Collections.Generic;
using System.Text;

namespace _001_test_工厂模式
{
    class ConcreteCreator2 : ConCretor
    {
        public override IProduct FactoryMethod()
        {
            return new ConcreteProduct2();
        }
    }
}

Client

using System;
using System.Collections.Generic;
using System.Text;

namespace _001_test_工厂模式
{
    public class Client
    {

        public void Main()
        {
            Console.WriteLine("通过客户端中的ConCretor类调用子类接口中的方法,调用了具体产品1的创建者方法");
            ClientCode(new ConcreteCreator1());

            Console.WriteLine("");

            Console.WriteLine("通过客户端中的ConCretor类调用子类接口中的方法,调用了具体产品2的创建者方法");
            ClientCode(new ConcreteCreator2());
        }

        //客户端代码与具体创建者的实例一起工作,尽管是通过其基本接口。只要客户端通过基接口继续与创建者工作,您就可以将创建者的任何子类传递给它。
        public void ClientCode(ConCretor creator)
        {
            // ...
            //客户端:我不知道创建者的类是“+”但它仍然有效。
            Console.WriteLine("客户端:我不知道ConCretor类是一个什么样的类,但是我依旧工作\n" + creator.SomeOperation());
            //Console.WriteLine("调用子类的产品流水线");
            //creator.RuneLine();//统一调用所有子类的流水线方法

            //如果说需要针对不同类型的产品类型做处理,我们也一样可以通过switch case做处理
            if(typeof(ConcreteCreator1)== creator.GetType())
            {
                Console.WriteLine("此时创建的类型为 具体产品类1");
            }
            else
            {
                Console.WriteLine("此时创建的类型不是具体产品类1");
                creator.RuneLine();

            }

            //还有一种思路,我们是不是可以尝试去对这个进行switch case的转换?来简化if的操作呢?但是我们也要注意,一个注重程序代码质量和美感的程序员
            //都在为了减少switch casse的使用,我们称之为代码整洁,所以我们要根据对应的情况去使用,如果说成千上万个case杂糅在一起是否是一种好的操作呢?
            //这点需要我们进行思考;
            // ...
        }
    }
}

Program

using _001_test_工厂模式;
using System;

namespace _001_工厂模式_test
{
    class Program
    {
        static void Main(string[] args)
        {
            new Client().Main();
        }
    }
}

执行结果:

①产品内部执行的方法和流程都一致,调用所有产品类的方法:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dGUs3j81-1669545517115)(C:\Users\JackiieWang\AppData\Roaming\Typora\typora-user-images\image-20221127183154708.png)]

②当不同的产品内部发生变化时,我们需要根据不同的产品类型执行不同的操作时:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gMFucpDu-1669545517115)(C:\Users\JackiieWang\AppData\Roaming\Typora\typora-user-images\image-20221127183244743.png)]

1.7 游戏开发中的应用

作为一个游戏开发者,我相信大家对这个更是广为应用;而对于我而言,我的理解:

像我们常见玩的游戏里面都有一种技能或者同类型不同操作的功能,我们是否可以尝试用工厂模式来使代码尽可能的减少与父类之间耦合,达到单一职责的目的;

例如:技能常见的分为 主动技能,被动技能,增益技能,技能有待学习,技能已经学习,技能可以升级,技能已经满级,技能的等级状态等等;

我们都可以去思考是否可以通过这种途径去做处理;

以上便是我对这个模式的了解和学习总结,希望能够帮助大家去理解这个设计模式,当然如果发现我的讲解其实是有问题的,欢迎大家指出我的问题所在,也欢迎大家留言讨论,设计模式的学习一直在路上,我后续还是跟进设计模式的理解和学习,想要了解的朋友可以关注我哈~嘻嘻(●’◡’●)
在这里插入图片描述
以上呢,便是我今日学习并实践工厂模式的总结,希望能够对你有所帮助~ 也希望你能够点赞、评论吖~ 你们的点赞、评论就是我前进的动力!
在这里插入图片描述
作者:ProMer_Wang

链接:https://blog.csdn.net/qq_43801020/article/details/128067846

本文为ProMer_Wang的原创文章,著作权归作者所有,转载请注明原文出处,欢迎转载!

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

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

相关文章

领悟《信号与系统》之 非周期信号的傅里叶变换

非周期信号的傅里叶变换一、非周期信号的傅里叶变换二、 典型信号的傅立叶变换1.单边指数信号2.偶双边指数3. 矩阵脉冲信号4. 奇双边指数5. 符号函数6. 冲激信号7. 阶跃信号三、常用傅里叶变换表这里记录的信号都是非周期信号的傅里叶变化,频谱变换的特点就是&#…

【计算机网络】数据链路层:使用广播信道的数据链路层(1)

局域网的数据链路层 局域网特点:网络为一个单位所拥有,地理范围和站点数目均有限。 地理范围和站点数目均有限。 局域网优点: 具有广播功能,从一个站点可以很方便地访问全网。 便于系统的拓展和演变,各设备的位置…

Discourse 论坛激活邮件问题

根据 Discourse 的官方推荐,我们使用的是 MailGun 的服务。 在大部分情况下都没有问题,但是在一些特定的邮件地址,例如 iCloud,我们在发送激活邮件的时候有提示为: "message": "5.5.1 Error: need MA…

实验:温湿度数据oled显示

OK,本次介绍一个oled实验 本来只想做oled实验的 后面想想这个实验太简单 就加上了温湿度传感器 oled可以打印英文和数字,比如用display.println(“Hello World!”)就可以了 如果打印汉字就比较复杂了 需要相应的软件,生成编码 然后一个字一个字打印 不过只要汉字不…

【数据结构】树——二叉树

1.树的介绍以及树的基本概念和性质 2.二叉树介绍以及二叉树的性质 3.二叉树的构建:穷举创建,递归创建 4.二叉树的基本操作 之前我们介绍了顺序表,链表,以及栈和队列,这几种数据结构都属于线性结构,而我们接…

GreenPlum/PostGreSQL表锁处理

GreenPlum/PostGreSQL表锁处理 数据库中遇到表锁的情况,可以通过select * from pg_stat_activity;查看表锁的进程及进程ID,从而取消进程,解锁。 一、模拟表锁 1.1 模拟表数据 创建lock_test表,并随意插入一条数据,…

Oracle自治事务示例演示

自治事务 自治事务(Pragma autonomous_transaction):是PL/SQL块中的一个单独事务,与调用或触发自己的事务之间互不干扰,自己commit和rollback不会影响其他事务,也不会被其他事务所影响。 通俗的讲&#xff…

【Android App】在线语音识别功能实现(使用云知声平台与WebSocket 超详细 附源码)

需要源码和相关资源请点赞关注收藏后评论区留下QQ~~~ 一、在线语音识别 云知声的语音识别同样采用WebSocket接口,待识别的音频流支持MP3和PCM两种格式,对于在线语音识别来说,云知声使用JSON串封装报文,待识别的音频以二进制形式发…

webpack打包vue文件+gulp打包sass文件

webpack打包vue文件 1,下载依赖 npm i vue-loader npm i webpack-cli2,编写webpack配置文件 /*** 关于webpack的配置文件*/const path require(path)const { VueLoaderPlugin } require(vue-loader)const glob require(glob) // node自带的读取文件的库 /*** …

会多门编程语言的你,最推荐哪3-5门语言?

如果你还想在编程的路上继续提高,那我建议你至少学习4种编程语言。可用的编程语言有很多,所以选择一种感兴趣的学习就可以了。我这么建议的原因是,要掌握编程,建立信心,提高能力,最简单的办法就是学习多种编…

浅析工作流调度器Azkaban

title: Azkaban系列 第一章 概述 1.1 为什么需要工作流调度器 1、一个完整的数据分析系统通常都是由大量任务单元组成: shell 脚本程序,java 程序,mapreduce 程序、hive 脚本等 2、各任务单元之间存在时间先后及前后依赖关系 3、为了很好地…

TIA西门子博途V18安装教程及注意事项

TIA西门子博途V18安装教程及注意事项 前提条件: TIA Portal V18需要.Net Framework 3.5环境,所以在安装TIA V18之前要先安装它。大家可以在控制面板中的程序和功能中检查是否已经安装,如果没有,可以参考以下步骤自行安装: 操作系统&#x

jsp旅行网系统Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 JSP 旅行网系统 是一套完善的web设计系统,对理解JSP java编程开发语言有帮助,系统具有完整的源代码和数据库,系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Myeclipse8.5开发,数据库为Mysql,使用java语…

[附源码]Python计算机毕业设计房屋租赁系统

项目运行 环境配置: Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术: django python Vue 等等组成,B/S模式 pychram管理等等。 环境需要 1.运行环境:最好是python3.7.7,…

2022级浙大MEM录取经验过程分享——有需求就要去匹配

我是2022 级 浙大MEM 新录取的考生,去年联考初试取得了213 分的成绩,综合拿到了148分,去年的提面中也拿到了优秀资格,在备考的过程中自己的一些心得体会和经验分享给大家,希望能够有所帮助。我的本科其实比较普通&…

python循环中的continue和break

目录 一.python中的continue 案例1 结果是 注意 案例2 结果是 二.python中的break 案例1 结果是 注意 案例2 结果是 三. python中continue和break的总结 一.python中的continue continue关键字用于:中断本次循环,直接进入下一次循环 continue可以用于:fo…

2.1.2 运算放大器的组成与分类、运算放大器的发展历程

笔者电子信息专业硕士毕业,获得过多次电子设计大赛、大学生智能车、数学建模国奖,现就职于南京某半导体芯片公司,从事硬件研发,电路设计研究。对于学电子的小伙伴,深知入门的不易,特开次博客交流分享经验&a…

1-10嵌入式Linux系统开发与应用|嵌入式Linux|第三章 Linux编程环境|下篇

目录 1.gcc编译器的使用 1.1gcc软件包 1.2一个基本实例 1.3gcc的主要选项 1.3.1指定函数库和包含文件的查找路径 1.3.2出错检查及警告 1.3.3优化选项 优化带来的问题 1.3.4调试选项 2.GNU C扩展简介 3.GNU make管理项目 3.1make简介 使用make管理项目的原因 4.编…

Nginx那些事儿2

负载均衡 当访问的服务具有多个实例时,需要根据某种“均衡”的策略决定请求发往哪个节点,这就是所谓的负载均衡,目的是为了将数据流量分摊到多个服务器执行,减轻每台服务器的压力,从而提高了数据的吞吐量 负载均衡的种类 常见的硬件有NetScaler、F5、Radware和Array等商用的负…

读懂英文文章所需的单词量

简介 备考托福,GRE需要背上万单词,除去考试通关的因素,就想看看是不是真有必要花时间去背那么多单词。 实验使用从初中到GRE不同等级考试要求的单词表,代入Brown文本数据集,评估背会各等级单词后,能看懂多…