面向对象设计模式纵横谈:Factory Method 工厂方法模式(笔记记录)
从耦合关系谈起
耦合关系直接决定着软件面对变化时的行为
-模块与模块之间的紧耦合使得软件面对变化时,相关模块都要随之更改
-模块与模块之间的松耦合使得软件面对变化时,一些模块更容易被替换或者更改,但其他模块保持不变
抽象部分变化慢,细节(具体)部分变化快;高层部分变化慢,底层部分变化快。
当我们对于系统的认识无法梳理出上面的图时,最好不要一开始就用设计模式,设计模式其实是一个演绎的过程。当我们对软件认识不断深化时,慢慢就会知道哪些是主要的,哪些是次要的,就能梳理出一个抽象和具体的层次,再考虑用哪种设计模式。
第二幅图满足了依赖倒置原则,中间的主线是变化慢的部分,分支都是用接口相连。这样的松耦合使得模块与模块之间的连接用接口连接,接口是相对稳定的部分,接口的实现是相对变化的部分。例如:去帮我买一条毛巾,只告诉了是毛巾这样事物,而毛巾的具体品种、颜色并没有具体的需求。大自然创造的世界,遍地都是松耦合、高内聚。例如屋子里的凳子和桌子、床单和被子等,当我们需要换床单时,是不需要换床的。床和床单之间有一个接口,床是主线,床需要床单的接口,只要具体的床单满足这个尺寸接口,就可以接上。主逻辑的变化成本比辅逻辑的变化成本高,所以尽量让辅逻辑的变化较少的影响主逻辑。因此我们设计软件的原则是,先稳定下接口,再考虑具体实现。
动机(Motivation)
在软件系统创建过程中,经常面临着“某个对象”的创建工作:由于需求的变化,这个对象(的具体实现)经常面临着剧烈的变化,但是它却拥有比较稳定的接口。
如何应对这种变化?如何提供一种“封装机制”来隔离出“这个易变对象”的变化,从而保持系统中“其他依赖对象的对象”不随着需求改变而改变?
意图(Intent)
定义一个用于创建对象的接口,让子类决定实例化哪一个类。Factory Method使得一个类的实例化延迟到子类。——《设计模式》GoF
结构(Structure)
例说Factory Method应用
汽车
汽车测试
但是这种测试只能测试一种Car,如果要测试其他类型的Car,需要修改代码并重新编译。
为了应对这种改变,我们需要把Car先变成抽象类。
然后我们在客户程序使用的时候,把所有的Car都换成抽象的AbstractCar,这样客户程序就不需要了解具体测试的是哪个Car了。客户程序如下
但这种代码明显是错误的,抽象类不能直接实例化,因此我们比较好的方法是把抽象的Car传递进来
如果现在我们需要Car的多个实例,那么参数只接收一个抽象的Car就显得不那么适用了。我们可能想到的方法是把传进来的Car做一个浅拷贝Memberwise,但是浅拷贝是一个protected方法,并且不能拷贝引用,但是这也是有办法解决的,这种克隆的做法可以,但是现在我们研究另一种做法。我们希望能有一个创建Car的工厂,这样我们在客户程序就不需要关心Car的实例,只管用这个工厂去创建具体的Car。
Car的具体实现
Car工厂
(纠正:CarFactory类中的CreateCar方法应该返回抽象的AbstractCar类型,CarFactory的类和具体HongqiCarFactory的类在实际中应该放在两个不同的文件夹中)
因为客户程序需要用到CarFactory,所以CarFactory中不应该涉及到Car的具体实现,CarFactory应该是一个抽象的工厂类,因此HongqiCar的工厂需要一个继承自抽象CarFactory的具体工厂。在应用程序调用的时候,传入客户程序的工厂应该是具体的HongqiCarFactory工厂。
当想换具体Car的时候,只需要创建一个新的Car继承自AbstractCar,并新建一个具体CarFactory工厂继承自抽象CarFactory。然后在具体的应用中把具体的Car工厂参数修改即可。当然,完全可以让具体应用的代码也不用修改,把变化转嫁到配置文件中去。
Factory Method模式的几个要点
Factory Method模式主要用于隔离类对象的使用者和具体类型之间的耦合关系。面对一个经常变化的具体类型,紧耦合关系会导致软件的脆弱。
Factory Method模式通过面向对象的手法,将所要创建的具体对象工作延迟到子类,从而实现一种扩展(而非更改)的策略,较好地解决了这种紧耦合关系。
Factory Method模式解决“单个对象”的需求变化;
AbstractFactory模式解决“系列对象”的需求变化;
Builder模式解决“对象部分”的需求变化;
.NET框架中的Factory Method应用
SOAP、ASP.Net HttpHandler等
面向对象设计模式纵横谈:Factory Method 工厂方法模式(笔记记录)的更多相关文章
- Java设计模式:Factory Method(工厂方法)模式
概念定义 工厂方法(Factory Method)模式,又称多态工厂(Polymorphic Factory)模式或虚拟构造器(Virtual Constructor)模式.工厂方法模式通过定义工厂抽 ...
- Factory Method 工厂方法模式
Factory method工厂方法模式是一种实现了“工厂”概念的面向对象设计模式.就像其他创建型模式一样,它也是处理在不指定对象具体类型的情况下创建对象的问题.工厂方法模式的实质是“定义一个创建对象 ...
- 设计模式C++学习笔记之五(Factory Method工厂方法模式)
工厂方法模式的意义是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类当中.核心工厂类不再负责产品的创建,这样核心类成为一个抽象工厂角色,仅负责具体工厂子类必须实现的接口,这样进一步抽象化的 ...
- 四、Factory Method 工厂方法模式
需求:对象的创建由工厂统一创建,会使用到模板模式 代码清单: 工厂: public abstract class Factory { public abstract Product createPro ...
- C#面向对象设计模式纵横谈——5.Factory Method 工厂方法模式(创建型模式)
动机 (Motivation) 在软件系统中,经常面临着“某个对象”的创建工作; 由于需求的变化,这个对象经常面临着剧烈的变化,但是它却拥有比较稳定的接口. 如何应对这种变化?如何提供一种“封装机制” ...
- 3.Factory Method 工厂方法模式(创建型模式)
1.定义: 定义一个用于创建对象的接口,让子类决定实例化哪一个类.Factory Method使得一个类的实例化延迟到子类. 2.实现代码如下: /// <summary> /// 工厂方 ...
- Factory Method工厂方法模式
定义一个用于创建对象的接口,让子类决定将哪一个类实例化.Factory Method使一个类的实例化延迟到其子类,属于创建型模式 在此模式中,工厂父类负责定义创建产品对象的公共接口,而工厂子类负责生产 ...
- 设计模式之 Factory Method 工厂方法
看到的比较有意思的一篇描述工厂方法的文章. http://www.codeproject.com/Articles/492900/From-No-Factory-to-Factory-Method 总 ...
- 设计模式04: Factory Methord 工厂方法模式(创建型模式)
Factory Methord 工厂方法模式(创建型模式) 从耦合关系谈起耦合关系直接决定着软件面对变化时的行为 -模块与模块之间的紧耦合使得软件面对变化时,相关的模块都要随之变更 -模块与模块之间的 ...
随机推荐
- 并发工具类(四)线程间的交换数据 Exchanger
前言 JDK中为了处理线程之间的同步问题,除了提供锁机制之外,还提供了几个非常有用的并发工具类:CountDownLatch.CyclicBarrier.Semphore.Exchanger.Ph ...
- django-重写User模型
User模型有很多功能,验证什么的,重写需要满足下面的功能(基本上写注释的地方都是需要的) 开始: 创建一个重写user的app, 记得注册app startapp newauth from djan ...
- TCP keep-alive翻译
原文链接: http://www.freesoft.org/CIE/RFC/1122/114.htmhttp://www.freesoft.org/CIE/RFC/1122/index.htm 实现T ...
- 练手nginx反向代理和负载均衡apache实战
先说下原理性的 什么是反向代理 用户访问域名 域名的指向到nginx nginx把请求转发到apache apache处理后 返回给用户 整套的逻辑 对于用户来说 就是访问域名 然后返回 没 ...
- ASP.NET CMS: Administration Template
ASP.NET CMS: Administration Template For many creating advanced ASP.NET website or application admin ...
- JAVA Spring 简单的配置和操作 ( 创建实体类, 配置XML文件, 调试 )
< 1 > 实体类 Person package java_spring.modle; /** * 一个实体类( Person ) */ public class Person { pri ...
- eclipse 的project explorer问题,这个怎样把localFileSystem去掉
转自:https://zhidao.baidu.com/question/550279043.html
- IdUDPServer 收到4次重复的数据
IdUDPServer1->Send(RemoteIP, LabeledEdit2->Text.ToInt(), InText, IndyTextEncoding_UTF8()); 我发给 ...
- J2SE 8的输入输出--Path/Paths File/Files; FileSystems 类的用法
Path的简单用法 //1. Path 正常用法 Path path = Paths.get("src/main/resource/zip"); logger.debug(path ...
- cacti客户端snmp设置
1. ubuntu : apt-get install snmp snmpd vim /etc/default/snmpd //将此配置文件中127.0.0.1 删掉. /etc/init.d/sn ...