用C#(.NET Core) 实现简单工厂和工厂方法设计模式

本文源自深入浅出设计模式. 只不过我是使用C#/.NET Core实现的例子.

 

前言

当你看见new这个关键字的时候, 就应该想到它是具体的实现.

640?wx_fmt=png&wxfrom=5&wx_lazy=1

这就是一个具体的类, 为了更灵活, 我们应该使用的是接口(interface).

有时候, 你可能会写出这样的代码:

640?wx_fmt=png&wxfrom=5&wx_lazy=1

这里有多个具体的类被实例化了, 是根据不同情况在运行时被实例化的. 

当你看到这样的代码, 你就会知道当有需求需要对其进行修改或者扩展的时候, 你就得把这个文件打开, 然后看看在这里应该添加或者删除点什么. 这类的代码经常会分散在程序的多个地方, 这维护和更新起来就很麻烦而且容易出错.

针对interface进行编程的时候, 你知道可以把自己独立于系统未来可能要发生的变化. 为什么呢? 因为如果你针对interface编程, 那么对于任何实现了该接口的具体类对你来说都可以用, 多态吗.

项目原始需求

有一个前沿的披萨店, 做披萨, 下面是订购披萨的类:

640?wx_fmt=png

new一个披萨, 然后按照工序进行加工 最后返回披萨.

但是, 一个披萨店不可能只有一种披萨, 可能会有很多中披萨, 所以你可能会这样修改代码:

640?wx_fmt=png

根据传入的类型, 创建不同的披萨, 然后加工返回.

然后问题来了, 随着时间的推移, 一个披萨店会淘汰不畅销的披萨并添加新品种披萨.

使用上面的代码就会出现这个问题, 针对需求变化, 我不得不把OrderPizza的部分代码改来改去:

640?wx_fmt=png

从这里, 我们也可以看到, 上半部分是会变化的部分, 下半部分是不变的部分, 所以它们应该分开(把变化的部分和不变的部分分开, 然后进行封装).

结构应该是这样的:

640?wx_fmt=png

右上角是变化的部分, 把这部分封装到一个对象里, 它就是用来创建披萨的对象, 我们把这个对象叫做: 工厂.

工厂负责创建对象的细节工作. 我们创建的这个工厂叫做SimplePizzaFactory, 而orderPizza()这个方法就是该工厂的一个客户(client).

任何时候客户需要披萨的时候, 披萨工厂就会给客户创建一个披萨.

接下来, 我们就建立这个简易的披萨工厂:

640?wx_fmt=png

就是通过传入的类型参数, 建立并返回不同类型的披萨.

这样我们就把披萨创建的工作封装到了一个类里面, 发生变化的时候, 只需要修改这一个类即可.

注意: 有时候上面这种简单工厂可以使用静态方法, 但是这样也有缺点, 就是无法通过继承来扩展这个工厂了.

回来修改PizzaStore这个类:

640?wx_fmt=png

工厂是从构造函数传入的, 并在PizzaStore里面保留一个引用.

在OrderPizza()方法里面, 我们使用工厂的创建方法代替了new关键字, 所以在这里没有具体的实例化.

简单工厂的定义

简单/简易工厂并不是一个设计模式, 更多是一个编程习惯. 但是使用的非常广泛.

简单工厂类图:

640?wx_fmt=png

这个很简单, 就不解释了. 

简单工厂就到这, 下面要讲两个重量级的工厂模式.

用C#/.NET Core实现简单工厂

Pizza父类:

using System;

using System.Collections.Generic;


namespace SimpleFactory.Pizzas

{

    public abstract class Pizza

    {

        public string Name { get; protected set; }

        public string Dough { get; protected set; }

        public string Sauce { get; protected set; }

        protected List<string> Toppings = new List<string>();


        public void Prepare()

        {

            Console.WriteLine($"Preparing: {Name}");

            Console.WriteLine($"Tossing: {Dough}");

            Console.WriteLine($"Adding sauce: {Sauce}");

            Console.WriteLine("Adding toppings: ");

            Toppings.ForEach(x => Console.WriteLine($"  {x}"));

        }


        public void Bake()

        {

            Console.WriteLine("Bake for 25 minutes");

        }


        public void Cut()

        {

            Console.WriteLine("Cutting the pizza into diagnol slices");

        }


        public void Box()

        {

            Console.WriteLine("Placing pizza in official PizzaStore box......");

        }

    }

}

各种Pizza:

namespace SimpleFactory.Pizzas

{

    public class CheesePizza: Pizza

    {

        public CheesePizza()

        {

            Name = "Cheese Pizza";

            Dough = "Think Dough";

            Sauce = "Salad";

            Toppings.Add("Grated Reggiano Cheese");

        }

    }

}


namespace SimpleFactory.Pizzas

{

    public class ClamPizza: Pizza

    {

        public ClamPizza()

        {

            Name = "Clam Pizza";

            Sauce = "Tomato sauce";

            Dough = "Soft dough";

            Toppings.Add("Shrimp meat");

        }

    }

}


namespace SimpleFactory.Pizzas

{

    public class PepperoniPizza: Pizza

    {

        public PepperoniPizza()

        {

            Name = "Pepperoni Pizza";

            Dough = "Thin dough";

            Sauce = "Black pepper";

            Toppings.Add("Beef Granules");

            Toppings.Add("Niblet");

        }

    }

}

简单工厂:

using SimpleFactory.Pizzas;


namespace SimpleFactory

{

    public class SimplePizzaFactory

    {

        public Pizza CreatePizza(string type)

        {

            Pizza pizza = null;

            switch (type)

            {

                case "cheese":

                    pizza = new CheesePizza();

                    break;

                case "pepperoni":

                    pizza = new PepperoniPizza();

                    break;

                case "clam":

                    pizza = new ClamPizza();

                    break;

            }


            return pizza;

        }

    }

}


PizzaStore:

using SimpleFactory.Pizzas;


namespace SimpleFactory

{

    public class PizzaStore

    {

        private readonly SimplePizzaFactory _factory;


        public PizzaStore(SimplePizzaFactory factory)

        {

            _factory = factory;

        }


        public Pizza OrderPizza(string type)

        {

            var pizza = _factory.CreatePizza(type);

            pizza.Prepare();

            pizza.Bake();

            pizza.Cut();

            pizza.Box();

            return pizza;

        }

    }

}

测试运行:

using System;


namespace SimpleFactory

{

    class Program

    {

        static void Main(string[] args)

        {

            var pizzaStore = new PizzaStore(new SimplePizzaFactory());

            var cheesePizza = pizzaStore.OrderPizza("cheese");

            Console.WriteLine();

            var clamPizza = pizzaStore.OrderPizza("pepperoni");

            Console.ReadKey();

        }

    }

}

640?wx_fmt=png

 

需求变更 - 授权连锁披萨店

 披萨店开的很好, 所以老板在全国各地开授权连锁分店了, 而每个地点的分店根据当地居民的口味, 它们所提供的披萨种类可能会不同.

例如纽约和芝加哥和加利福尼亚的就有可能不同.

针对这个需求, 我们可能会想到的第一种办法就是: 把SimplePizzaFactory抽取出来, 分别建立三个地点的工厂, 然后根据地点把相应的工厂组合到PizzaStore

640?wx_fmt=png

代码是这样的:

纽约:

640?wx_fmt=png

芝加哥:

640?wx_fmt=png

因为个连锁店分布在各地, 老板想做质量管控: 做披萨的基本工序应该是一样的, 但是针对某种披萨各地可以有不同的风格做法.

所以我们把createPizza()方法放回到PizzaStore, 但这次它是抽象方法, 然后各地都会创建自己的PIzzaStore:

640?wx_fmt=png

下面是纽约和芝加哥的披萨店:

640?wx_fmt=png

针对每种披萨, 纽约和芝加哥可能会有自己风格具体实现的披萨.

orderPizza()方法是在父类/抽象类里面实现的, 这里的披萨还是抽象的, 所以它并不知道是PizzaStore的哪个子类来做的披萨.

代码运行的时候, orderPizza()会调用createPizza()方法, PizzaStore的某个子类肯定会对此负责.

所以你哪个地方的PizzaStore, 就会决定产出的是哪个地方特产的披萨.

 

下面就创建PizzaStore, 例如纽约的:

640?wx_fmt=png

其他地点的都差不多, 就不贴图了.

如何声明一个工厂方法

还是看这张图:

640?wx_fmt=png

抽象的PizzaStore把订购披萨的固定工序orderPizza()放在了抽象类里面.

创建披萨createPizza()方法是在各地的披萨店里做实现.

用一行代码来解释工厂方法就是:

640?wx_fmt=png

工厂方法是让其子类具体来实现对象创建的工作. 这样就把父类中的客户代码和子类的创建对象部分的代码解耦了.

 

上面工作做的挺好, 但是还差一件事....披萨.

首先抽象父类:

640?wx_fmt=png

里面定义了调味料和工序

然后具体的披萨:

纽约的奶酪披萨

640?wx_fmt=png

芝加哥的奶酪披萨

640?wx_fmt=png

最后运行一下:

640?wx_fmt=png

640?wx_fmt=png

 

工厂方法模式

所有的工厂模式都会封装对象的创建过程, 而工厂方法模式把对象创建的动作交给了子类, 并让它决定创建哪些对象.

创建者:

640?wx_fmt=png

 

产品:

640?wx_fmt=png

看看另外一种结构 -- 并行的类结构:

640?wx_fmt=png

 

工厂方法模式的定义:

工厂方法模式定义了一个创建对象的接口, 但是让子类来决定具体创建的是哪一个对象. 工厂方法让一个类延迟实例化, 直到子类的出现.

640?wx_fmt=png

左边是产品, 所有具体的产品都应该继承于同一个父类/接口.

右边的Creator类里面包含所有方法的实现除了抽象的工厂方法. 这个抽象的工厂方法在Creator的子类里面必须进行实现, 产品就是在子类具体实现的工厂方法里面创造出来的.

 

设计原则 -- 应该依赖于抽象, 而不依赖于具体的类

这就是著名的: DIP (Dependency Inversion Principle) 依赖反转原则.

进一步解释就是: 高级别的组件不应该依赖于低级别的组件, 它们都应该依赖于抽线.

高级别组件, 就是它有一组行为定义在另外一堆低级别的组件里面了.

例如PizzaStore就是高级别的, 具体的披萨就是低级别的.

应该该设计原则后:

640?wx_fmt=png

这时它们都依赖于抽象的披萨父类了.

实现该原则的三点指导建议

  • 没有变量引用具体的类(可已使用工厂代替创建这个具体的类)

  • 没有类派生于具体的类(派生于它就依赖于它)

  • 不去重写(override)其任一父类的已实现方法(如果重写了, 那么这个类并不适合作为起始的抽象类, 因为基类里面的方法本应该是共享与所有子类的)

和其它原则一样, 只是尽力去按照这三点建议去执行, 并不是必须一直要这么做.

C#/.NET Core的代码实现

各种pizza:

namespace FactoryMethodPattern.Pizzas

{

    public class ChicagoCheesePizza : Pizza

    {

        public ChicagoCheesePizza()

        {

            Name = "Chicago Cheese Pizza";

            Dough = "Think Dough 1";

            Sauce = "Salad 1";

            Toppings.Add("Grated Reggiano Cheese 1");

        }

    }

}


namespace FactoryMethodPattern.Pizzas

{

    public class ChicagoClamPizza : Pizza

    {

        public ChicagoClamPizza()

        {

            Name = "Chicago Clam Pizza";

            Sauce = "Tomato sauce 1";

            Dough = "Soft dough 1";

            Toppings.Add("Shrimp meat 1");

        }

    }

}


namespace FactoryMethodPattern.Pizzas

{

    public class ChicagoPepperoniPizza : Pizza

    {

        public ChicagoPepperoniPizza()

        {

            Name = "Chicago Pepperoni Pizza";

            Dough = "Thin dough 1";

            Sauce = "Black pepper 1";

            Toppings.Add("Beef Granules 1");

            Toppings.Add("Niblet 1");

        }

    }

}


namespace FactoryMethodPattern.Pizzas

{

    public class NYCheesePizza: Pizza

    {

        public NYCheesePizza()

        {

            Name = "NY Cheese Pizza";

            Dough = "Think Dough 2";

            Sauce = "Salad 2";

            Toppings.Add("Grated Reggiano Cheese 2");

        }

    }

}


namespace FactoryMethodPattern.Pizzas

{

    public class NYClamPizza: Pizza

    {

        public NYClamPizza()

        {

            Name = "NY  Clam Pizza";

            Sauce = "Tomato sauce 2";

            Dough = "Soft dough 2";

            Toppings.Add("Shrimp meat 2");

        }

    }

}


namespace FactoryMethodPattern.Pizzas

{

    public class NYPepperoniPizza: Pizza

    {

        public NYPepperoniPizza()

        {

            Name = "NY Pepperoni Pizza";

            Dough = "Thin dough 2";

            Sauce = "Black pepper 2";

            Toppings.Add("Beef Granules 2");

            Toppings.Add("Niblet 2");

        }

    }

}

披萨店抽象父类:

using FactoryMethodPattern.Pizzas;


namespace FactoryMethodPattern

{

    public abstract class PizzaStore

    {

        public Pizza OrderPizza(string type)

        {

            var pizza = CreatePizza(type);

            pizza.Prepare();

            pizza.Bake();

            pizza.Cut();

            pizza.Box();

            return pizza;

        }


        protected abstract Pizza CreatePizza(string type);

    }

}

Chicago披萨店:

using FactoryMethodPattern.Pizzas;


namespace FactoryMethodPattern

{

    public class ChicagoPizzaStore: PizzaStore

    {

        protected override Pizza CreatePizza(string type)

        {

            Pizza pizza = null;

            switch (type)

            {

                case "cheese":

                    pizza = new ChicagoCheesePizza();

                    break;

                case "pepperoni":

                    pizza = new ChicagoPepperoniPizza();

                    break;

                case "clam":

                    pizza = new ChicagoClamPizza();

                    break;

            }


            return pizza;

        }

    }

}

纽约披萨店:

using FactoryMethodPattern.Pizzas;


namespace FactoryMethodPattern

{

    public class NYPizzaStore : PizzaStore

    {

        protected override Pizza CreatePizza(string type)

        {

            Pizza pizza = null;

            switch (type)

            {

                case "cheese":

                    pizza = new NYCheesePizza();

                    break;

                case "pepperoni":

                    pizza = new NYPepperoniPizza();

                    break;

                case "clam":

                    pizza = new NYClamPizza();

                    break;

            }


            return pizza;

        }

    }

}


测试运行:

using System;


namespace FactoryMethodPattern

{

    class Program

    {

        static void Main(string[] args)

        {

            var nyStore = new NYPizzaStore();

            var chicagoStore = new ChicagoPizzaStore();


            var pizza = nyStore.OrderPizza("cheese");

            Console.WriteLine($"Ordered a {pizza.Name} in NY");

            Console.WriteLine();

            var pizza2 = chicagoStore.OrderPizza("cheese");

            Console.WriteLine($"Ordered a {pizza2.Name} in Chicago");


            Console.ReadKey();

        }

    }

}

640?wx_fmt=png

 

相关文章:

  • C#  观察者模式 以及 delegate 和 event

原文地址 http://www.cnblogs.com/cgzl/p/8760250.html


.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

640?wx_fmt=jpeg

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

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

相关文章

1、oracle数据库简介

1.ORACLE数据库简介 数据库指的是存储和管理数据的仓库&#xff0c;是一种文件集合&#xff08;包括数据文件、临时文件、日志文件和控制文件&#xff09;&#xff0c;我们一般所说的数据库指的是数据库管理系统&#xff0c;一种用于操作数据库的软件&#xff0c;简称DBMS。OR…

U102488-傻叉题【dp】

前言 %\%%一下出题人BPMBPMBPM 正题 题目链接:https://www.luogu.com.cn/problem/U102488 题目大意 nnn个数&#xff0c;选取两组(不一定要全用上)使得他们的和相等且最大。 解题思路 考虑dpdpdp&#xff0c;fi,jf_{i,j}fi,j​表示到第iii个&#xff0c;差值为jjj(这里的差值…

TypeScript 2.8引入条件类型

最新发布的TypeScript 2.8包含了若干主要特性和一些问题修复&#xff0c;其中最为重要的是新增了条件类型&#xff0c;开发人员可以根据其他类型的特征为变量选择适当的类型。条件类型最适合与泛型组合在一起使用。如果一个框架总是重复相同的选择性代码&#xff0c;就会变得很…

2、oracle数据库的用户和权限

ORACLE用户和权限 每个用户都有一个默认表空间和一个临时表空间&#xff0c;因此操作用户的顺序一般是先创建表空间&#xff0c;然后创建用户&#xff0c;之后赋权。 1.创建表空间 1.1 创建临时表空间 create temporary tablespace test_tmp tempfile /home/oracle/aracle…

jzoj6451-[2020.01.19NOIP提高组]不幸运数字【记忆化搜索,数位dp,高精度】

正题 题目链接:https://jzoj.net/senior/#main/show/6451 题目大意 给出a,ba,ba,b&#xff0c;求[a,b][a,b][a,b]这个区间中有多少数字包含444。 解题思路 用fi,j,kf_{i,j,k}fi,j,k​表示到第iii位&#xff0c;是否含有444&#xff0c;前面的位是否都到达上限。 然后dfsdfs…

用分布式缓存提升ASP.NET Core性能

得益于纯净、轻量化并且跨平台支持的特性&#xff0c;ASP.NET Core作为热门Web应用开发框架&#xff0c;其高性能传输和负载均衡的支持已广受青睐。实际上&#xff0c;10-20台Web服务器还是轻松驾驭的。有了多服务器负载的支持&#xff0c;使得Web应用层在业务增长时随时采用水…

3、oracle数据库的语法基础

ORACLE语法基础 ORACLE的语法基础主要包括数据类型和SQL语法。 1.数据类型 1.1字符型 用于存储文本的数据类型 CHAR&#xff08;固定长度字符串1-2000个字节&#xff09; VARCHAR2&#xff08;可变长度字符串&#xff0c;字符串1-4000个字节&#xff09; LONG&#xff08;更…

123记住密码设置

1、打开123所在安装目录 2、进入config目录 3、找到.ovpn文件 4、找到auth-user-pass&#xff0c;没有的话在最后填写即可 在.ovpn同级目录下创建一个.txt文件&#xff0c;然后在这里设置一下 例如&#xff1a;auth-user-pass pass.txt 5、创建.txt文件&#xff0c;填写用户名…

Visual Studio 2017 15.7预览版发布

Visual Studio 2017已经发布一年多了&#xff0c;微软一直持续定期推出更新。第7个预览版也已发布&#xff0c;这一版本继续带来大量的改进。首先是增加了对TypeScript 2.8的支持。TypeScript 2.8带来了条件类型、JSX Pragma和映射类型标识符的可控性。VS2017的其他改进还包括在…

P3377-[模板]左偏树(可并堆)

正题 题目链接:https://www.luogu.com.cn/problem/P3377 题目大意 开始时nnn个只有一个数的集合&#xff0c;要求支持 合并两个集合查询一个集合中的最小值并删除 解题思路 左偏树就是维护一个满足以下性质的树 对于valxval_xvalx​有valx<vallsxval_x<val_{ls_x}va…

4、oracle数据库的查询基础

ORACLE查询基础 介绍oracle下数据的查询时用到的一些运算符、关键字和函数。 1.运算符 在进行数据查询时&#xff0c;有的时候不只是查询字段值&#xff0c;还需要做一些数据处理的操作&#xff0c;这时候就用到了运算符 -- 算术运算符 - * / -- 连接运算符 || -- 比较运算…

2018年4月更新70多个公司dnc招聘职位

2018年4月更新70多个公司dnc招聘职位请在本页回复&#xff0c;补充dnc招聘信息、公司案例dnc简介dnc .NET Core、dotnet Core简写dnc是微软新一代主力编程平台&#xff0c;开源、免费、跨平台、轻量级、高性能&#xff0c;可部署到Linux、Docker、k8s等环境&#xff0c;适合开…

P1552-[APIO2012]派遣【左偏树】

正题 题目链接:https://www.luogu.com.cn/problem/P1552 题目大意 一个nnn个点森林&#xff0c;每个点有价值和代价&#xff0c;选择一个点并在这个点的子树中选择一些点使得。 选择的点数∗该点的价值选择的点数*该点的价值选择的点数∗该点的价值最大且选择的点的代价之和不…

5、oracle下数据完整性约束

ORACLE下数据完整性约束 为了保证数据的完整性和结构的正确性&#xff0c;oracle也有很多约束条件。 1.数据完整性 为了保证数据的完整性&#xff0c;一般对数据列&#xff08;字段&#xff09;进行如下约束&#xff0c;主键约束&#xff08;PRIMARY KEY&#xff09;、唯一键…

把旧系统迁移到.Net Core 2.0 日记(1) - Startup.cs 解析

因为自己到开发电脑转到Mac Air&#xff0c;之前的Webform/MVC应用在Mac 跑不起来&#xff0c;而且.Net Core 2.0 已经比较稳定了。1. 为什么会有跨平台的.Net Core 近年来&#xff0c;我们已经进入云计算时代&#xff0c;在云平台的PaSS和SaSS上也是发生了大幅度的进化&#x…

P3261-[JLOI2015]城池攻占【左偏树】

正题 题目链接:https://www.luogu.com.cn/problem/P3261 题目大意 nnn个点的树&#xff0c;每个节点有一个防御值和一个攻击后的影响(让你的伤害加上一个数或者乘上一个数) 然后mmm个骑士&#xff0c;给定初始攻击点和初始伤害&#xff0c;不停往上走&#xff0c;遇到防御小于…

6、oracle数据库下查询操作

ORACLE下查询操作 针对数据库操作最多的就是数据查询&#xff0c;这里分享一个我常用的方法&#xff0c;看到需求后&#xff0c;先确定查询范围&#xff0c;就是需要查询哪些表&#xff1b;之后确定查询条件&#xff1b;最后写出查询的字段。 ORACLE中之前说过有两个伪列ROWI…

ASP.NET Core 2.0 : 图说管道,唐僧扫塔的故事

本文通过一张GIF动图来继续聊一下ASP.NET Core的请求处理管道&#xff0c;从管道的配置、构建以及请求处理流程等方面做一下详细的研究。&#xff08;ASP.NET Core系列目录&#xff09;一、概述上文说到&#xff0c;请求是经过 Server监听>处理成httpContext>Application…

P3521-[POI2011]ROT-Tree【线段树合并】

正题 题目链接:https://www.luogu.com.cn/problem/P3521 题目大意 一棵二叉树&#xff0c;叶子节点有权值&#xff0c;对于每个非叶子节点可以选择交换左右节点&#xff0c;求最后遍历出来的叶子节点权值逆序对最少。 解题思路 十分显然一个节点是否交换是不影响该节点子树之…

7、oracle下的序列

ORACLE下的序列 序列是一种数据库对象&#xff0c;用于生成一系列的整数&#xff0c;可以用来唯一的标记一条记录&#xff0c;在mysql种有字段自增的概念&#xff0c;但是oracle种是没有字段自增的&#xff0c;所以可以使用序列来作为主键的自动生成方式。 1.序列的使用 序列…