设计模式中的那些工厂

设计模式中的那些工厂

Intro

设计模式中有几个工厂模式,聊一聊这几个工厂模式的各自用法和使用示例,工厂模式包含简单工厂,抽象工厂,工厂方法,这些均属于创建型模式, 所谓创建型模式,就是说这几个设计模式是用来创建对象的。

简单工厂

首先来说一说,最简单的简单工厂

简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例

严格的来说,简单工厂模式是工厂模式家族中最简单实用的模式,但不属于23种 GOF 设计模式之一。因为每次要新增类型的时候必须修改工厂内部代码,不符合开闭原则。

来看一个例子:

public class OperationFactory
{public static Operation CreateOperation(string operate){Operation operation = null;switch (operate){case "+":operation = new OperationAdd();break;case "-":operation = new OpertaionSub();break;case "*":operation = new OperationMul();break;case "/":operation = new OperationDiv();break;}return operation;}
}

这是一个简单的计算器的示例,支持简单的加减乘除操作,如果要增加一个操作的话就必须要有增加一个 switch ... case 分支,需要修改 CreateOperation 方法不能满足对扩展开放对修改关闭的开闭原则,所以普遍地认为简单工厂不属于设计模式之一,但是我觉得有时候简单的业务处理用简单工厂还是比较方便的。

抽象工厂

抽象工厂模式,提供一系列相关或相互依赖对象的接口,而无需指定他们具体的类。

实现抽象工作模式所需要的组件,主要部分:

  • 抽象工厂/抽象产品

  • 具体工厂1/具体产品1

  • 具体工厂2/具体产品2

  • ...

在客户端根据不同的配置选择不同的工厂,例如根据配置的数据库类型的不同选择使用 Access 数据库仓储的工厂还是使用 SqlServer 数据库的仓储工厂

示例:

IDbFactory factory = new AccessFactory();
var userRepo = factory.CreateUserRepo();
userRepo.Insert(null);
var departmentRepo = factory.CreateDepartmentRepo();factory = new SqlServerFactory();
userRepo = factory.CreateUserRepo();
userRepo.Insert(null);

工厂方法

工厂方法模式(Factory Method)定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到子类。

工厂方法模式实现时,客户端需要决定实例化哪一个工厂来实现客户端的操作,也会存在着选择判断的问题,不过和简单工厂相比,简单工厂的选择判断是在工厂内部,而工厂方法则将选择判断转移到了客户端。

示例:

ILeifengFactory factory = new UndergraduteFactory();
var studentLeifeng = factory.CreateLeifeng();
studentLeifeng.BuyRice();factory = new VolunteerFactory();
var leifeng1 = factory.CreateLeifeng();
leifeng1.Sweep();

More

工厂模式的作用无外乎下面这四个。这也是判断要不要使用工厂模式的最本质的参考标准。

  • 封装变化:创建逻辑有可能变化,封装成工厂类之后,创建逻辑的变更对调用者透明。

  • 代码复用:创建代码抽离到独立的工厂类之后可以复用。

  • 隔离复杂性:封装复杂的创建逻辑,调用者无需了解如何创建对象。

  • 控制复杂度:将创建代码抽离出来,让原本的函数或类职责更单一,代码更简洁。

工厂方法和抽象工厂的区别

工厂方法模式:定义一个用于创建对象的接口,让子类决定实例化哪一个类 抽象工厂模式:为创建一组相关或相互依赖的对象提供一个接口,而且无需指定他们的具体类 区别在于产品,如果产品单一,最合适用工厂模式,但是如果有多个业务品种、业务分类时,通过抽象工厂模式产生需要的对象是一种非常好的解决方式。再通俗深化理解下:工厂模式针对的是一个产品等级结构 ,抽象工厂模式针对的是面向多个产品等级结构的。

抽象工厂关键在于产品之间的抽象关系,所以一般至少要两个产品;工厂方法在于生成产品,不关注产品间的关系,所以可以只生成一个产品。

抽象工厂更像一个复杂版本的策略模式,策略模式通过更换策略来改变处理方式或者结果;而抽象工厂的客户端,通过更换工厂而改变结果。

工厂方法目的是生产产品,所以能看到产品,而且还要使用产品。当然,如果产品在创建者内部使用,那么工厂方法就是为了完善创建者,从而可以使用创建者。另外创建者本身是不能更换所生产产品的。

抽象工厂的工厂是类;工厂方法的工厂是方法。抽象工厂的工厂类就做一件事情生产产品。生产的产品给客户端使用,绝不给自己用。工厂方法生产产品,可以给系统用,可以给客户端用,也可以自己这个类使用。自己这个类除了这个工厂方法外,还可以有其他功能性的方法。

选择的优化

简单工厂因为选择是在工厂内部的,不符合开闭原则,抽象工厂和工厂方法是将选择权交给客户端,由客户端根据需要自己决定要实例化的工厂。在实际应用的时候大部分情况是只会使用一种工厂,这种情况我们一般可以借助反射+配置来优化选择,如果使用依赖注入,可以直接注入需要的服务即可。

使用反射+配置优化

private static readonly string AssemblyName = "AbstractFactoryPattern";
private static readonly string DbName = ConfigurationHelper.AppSetting("DbName");public static IUserRepo CreateUserRepo()
{return (IUserRepo)typeof(DataAccess).Assembly.CreateInstance($"{AssemblyName}.{DbName}UserRepo");
}public static IDepartmentRepo CreateDepartmentRepo()
{return (IDepartmentRepo)typeof(DataAccess).Assembly.CreateInstance($"{AssemblyName}.{DbName}DepartmentRepo");
}

使用依赖注入

依赖注入可以使得我们的代码变得更加良好,扩展性更强。

// 依赖注入
var builder = new ContainerBuilder();
builder.RegisterType<VolunteerFactory>().As<ILeifengFactory>();
builder.RegisterType<SqlServerFactory>().As<IDbFactory>();
var container = builder.Build();var leifengFactory = container.Resolve<ILeifengFactory>();
var volunteer = leifengFactory.CreateLeifeng();
volunteer.Wash();var dbFactory = container.Resolve<IDbFactory>();
dbFactory.CreateDepartmentRepo().CreateDepartment(null);

Reference

  • https://github.com/WeihanLi/DesignPatterns/blob/master/CreatePattern/SimpleFactoryPattern

  • https://github.com/WeihanLi/DesignPatterns/tree/master/CreatePattern/AbstractFactoryPattern

  • https://github.com/WeihanLi/DesignPatterns/blob/master/CreatePattern/FactoryMethodPattern

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

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

相关文章

[mybatis]映射文件_select_resultMap_关联查询

普通方式 public class Employee {private Integer id;private String lastName;private String email;private String gender;private Department dept;}public class Department {private Integer id;private String departmentName;}public interface EmployeeMapperPlus {…

使用sqlserver搭建高可用双机热备的Quartz集群部署

一般拿 Timer 和 Quartz 相比较的&#xff0c;简直就是对 Quartz 的侮辱&#xff0c;两者的功能根本就不在一个层级上&#xff0c;如本篇介绍的Quartz强大的集群机制&#xff0c;可以采用基于sqlserver&#xff0c;mysql的集群方案&#xff0c;当然还可以在第三方插件的基础上实…

从Memcached到Redis,从RabbitMQ到Kafka,为了高并发,这些年我们不容易!

周一上班又撕了一页台历&#xff0c;随即心里头却是一惊。如今高考落下帷幕&#xff0c;不曾想2020竟已过了一半&#xff01;这一年又是疫情又是洪水&#xff0c;我们的目光被各种重大事件所吸引&#xff0c;此时审视一下自己&#xff0c;突然觉得不踏实了&#xff01;一场疫情…

[mybatis]动态sql_if_where_trim判断OGNL

OGNL if 要求&#xff1a;查询员工&#xff0c;要求&#xff0c;携带了哪个字段查询条件就带上这个字段的值 //携带了哪个字段查询条件就带上这个字段的值public List<Employee> getEmpsByConditionIf(Employee employee);<!-- public List<Employee> getEm…

通过 Continual Learning 提高 ML.NET 模型准确性并增强性能

从事机器学习的学者大多认为持续学习&#xff08;Continual Learning&#xff09;是迈向人工智能的根本一步。持续学习是模型模仿人类在整个生命周期中不断从数据流中学习、微调、转移知识和技能的能力。实际上&#xff0c;这意味着模型在进入新数据时能自主学习和适应生产。过…

[mybatis]动态sql_choose_分支选择

choose 如果带了id就用id查&#xff0c;如果带了lastName就用lastName查&#xff1b;只会进入其中一个 public List<Employee> getEmpByConditionChoose(Employee employee);<!-- public List<Employee> getEmpByConditionChoose(Employee employee);-->…

7.15周三晚8点,dotnet课堂全新起航,张善友/陈计节/刘腾飞我们一起来聊聊abp的故事...

直播主题&#xff1a;我们和Abp的故事直播嘉宾&#xff1a;张善友&#xff0c;陈计节&#xff0c;刘腾飞直播话题张善友&#xff1a;我是如何使用Abp的刘腾飞&#xff1a;利用Abp的模块化解决单体和分布式混合架构陈计节&#xff1a;Abp开源项目的DevOps实践Abp VNext 处于被低…

[mybatis]动态sql_set_与if结合的动态更新

因为sql语句拼装后&#xff0c;set语句后面可能会多一个逗号&#xff0c;所以要用set标签 set <!-- public void updateEmp(Employee employee);--> <update id"updateEmp">update tb1_employee<set><if test "lastName!null"&…

ASP.NET Core端点路由 作用原理

端点路由(Endpoint Routing)最早出现在ASP.NET Core2.2&#xff0c;在ASP.NET Core3.0提升为一等公民。Endpoint Routing的动机在端点路由出现之前&#xff0c;我们一般在请求处理管道的末尾&#xff0c;定义MVC中间件解析路由。这种方式意味着在处理管道中&#xff0c;MVC中间…

[mybatis]动态sql_foreach_遍历集合批量插入

foreach遍历集合 collection:指定要遍历的集合 list类型的参数会特殊处理封装在map中&#xff0c;map的key就叫list item:将当前遍历出的元素赋值给指定的变量 #{变量名}就能取出变量的值也就是当前遍历出的元素 separator:每个元素之间的分隔符 open:遍历出所有结果拼接一…

程序员修神之路--分布式系统设计理念这么难学?

点击“蓝字”关注我们吧福利&#xff1a;有件小事想和大家说一下菜菜哥&#xff0c;问你个问题&#xff0c;为什么现在的系统都设计为分布式系统呢&#xff1f;这个问题问得好&#xff0c;就像为什么程序员会慢慢脱发一样神奇01PART分布式系统身为二十一世纪的一名程序员&#…

[mybatis]动态sql_sql_抽取可重用的sql片段

sql 抽取可重用的sql片段&#xff0c;方便后面引用 1.sql抽取&#xff1a;经常将要查询的列名&#xff0c;或者插入用的列名抽取出来方便引用2.include来引用已经抽取的sql3.include还可以自定义一些property&#xff0c;sql标签内部就能使用自定义属性 取值的正确方式&#x…

使用keepalived搭建双机热备高可用一览

很多时候大家为了部署高可用方案都是前端配一个 nginx&#xff0c;如果nginx挂掉怎么办&#xff0c;比如下面这张图&#xff1a;你可以清楚的看到&#xff0c;如果 192.168.2.100 这台机器挂掉了&#xff0c;那么整个集群就下线了&#xff0c;这个问题该怎么解决呢&#xff1f;…

Linus通过了Linux中避免master/slave等术语的提案

Linux 内核维护者 Dan Williams 曾于 7 月初提交一份提案&#xff0c;建议逐步取消 master/slave 和 blacklist/whitelist 术语。近日&#xff0c;Linus Torvalds 则在 Linux 5.8 版本库的拉取请求中批准了该提议。自此&#xff0c;Linux 开发人员则需要使用新的术语来替代 mas…

[mybatis]缓存_一级缓存_一级缓存失效的四种情况

1.sqlSession不同 Testpublic void test05() throws IOException {SqlSessionFactory sqlSessionFactory getSqlSessionFactory();SqlSession sqlSession01 sqlSessionFactory.openSession();try{EmployeeMapper mapper01 sqlSession01.getMapper(EmployeeMapper.class);Emp…

LINQ :最终统治了​所有的语言!

LINQ&#xff1a;最终统治了所有的语言&#xff01;让我们看看LINQ如何彻底改变了.NET中访问数据的方式.NET与其他技术栈的不同之处之一绝对是LINQ&#xff0c;它是Language Integrated Query的首字母缩写。实际上&#xff0c;它是随.NET Framework 3.5和Visual Studio 2008引入…

[SpringBoot2]文件上传_单文件与多文件上传的使用

<form role"form" th:action"{/upload}" method"post" enctype"multipart/form-data"><div class"form-group"><label for"exampleInputEmail1">邮箱</label><input type"email&…

使用Docker运行SQL Server

现在.net core已经跨平台了&#xff0c;大家也都用上了linux用上了docker。跟.net经常配套使用的SQL SERVER以前一直是windows only&#xff0c;但是从SQL Server 2017开始已经支持运行在docker上&#xff0c;也就说现在SQL Serer已经可以运行在linux下了。下面在Ubuntu 16.4上…

[mybatis]映射文件_select_resultMap_关联查询_association分步查询延迟加载

association分步查询 场景一 查询Employee的同时查询员工对应的部门 EmployeeDepartment 一个员工有与之对应的部门信息 Employee表: Department表&#xff1a; public interface DepartmentMapper {public Department getDeptById(Integer id);}public interface EmployeeMap…

C#中你想象的Task,很简单?

【导读】网上关于Task的文章如数家珍&#xff0c;不过有一部分并未谈到一个根本的问题&#xff0c;所创建的Task一定在线程池上运行&#xff1f;如何合理的使用Task&#xff1f;这里并不会去重新讲解每一个APi的使用&#xff0c;没有任何意义&#xff0c;这属于包括我在内的各位…