工厂方法模式

概述

       工厂方法模式,英文Factory method pattern,工厂方法模式是简单工厂模式的进化版, 看本文之间最好先看一下简单工厂模式。工厂方法模式是定义一个创建产品对象的工厂接口,工厂接口本身不去创建对象,而是交给其子类或者是其实现类去创建,将实际创建工作推迟到子类中进行,我们先看一下如何实现工厂方法模式,最后再说一下它有什么好处。

实现

       Operation.java,运算基类,抽象getResult方法。

public abstract class Operation {   protected double numA;   protected double numB;   public double getNumA() {      return numA;   }   public void setNumA(double numA) {      this.numA = numA;   }   public double getNumB() {      return numB;   }   public void setNumB(double numB) {      this.numB = numB;   }   public abstract double getResult();}

       AddOperation.java,加法运算类,继承自Operation,实现getResult方法。

/* 加法 */public class AddOperation extends Operation {   @Override   public double getResult() {      return numA + numB;   }}

       SubOperation.java,减法运算类,继承自Operation,实现getResult方法。

/* 减法 */public class SubOperation extends Operation {   @Override   public double getResult() {      return numA - numB;   }}

       乘法、除法运算类略,可以见简单工厂模式第四章第三节

       IFactory.java,创建运算类的工厂接口,包含一个创建运算类的抽象方法。

public interface IFactory {   Operation createOperation();}

       AddFactory.java,加法工厂,实现IFactory,创建加法运算类。

public class AddFactory implements IFactory {   public Operation createOperation() {      return new AddOperation();   }}

       SubFactory.java,减法工厂,实现IFactory,创建减法运算类。

public class SubFactory implements IFactory {   public Operation createOperation() {      return new SubOperation();   }}

       乘法、除法工厂类略。

       FactoryMethodTest.java,客户端类。

public class FactoryMethodTest {   public static void main(String[] args) {      IFactory factory = new AddFactory();      Operation operation = factory.createOperation();      operation.setNumA(1);      operation.setNumB(2);      System.out.println(operation.getResult());   }}

UML

       从UML类图中可以看出,工厂方法模式中工厂的个数是和运算类的个数是一样的,也就是说增加一个运算类,就需要增加一个工厂。而简单工厂方法模式中增加一个运算类,需要在静态工厂方法中增加一个判断分支,这样看起来好像工厂方法模式反而麻烦了。现在就得引入软件设计中的一个原则,开放-封闭原则。

原则

       软件设计中,有很多设计原则,可能我们并非一定要遵守,但我们一定要尽量遵守,因为这始终是利大于弊的。

开放-封闭原则

       针对扩展是开放的,针对修改是封闭的。详细介绍可以见百度百科,开放-封闭原则是面向对象设计的核心所在,遵循这个原则可以带来巨大好处,可扩展、可维护、可复用。再来说简单工厂方法模式,我需要增加一种运算时,需要增加一个新的运算类,这没有错,但是还需要修改工厂方法,增加分支,这就违背了开放-封闭原则,工厂方法模式就是基于其上的遵循开放-封闭原则的解决方案,针对工厂的变化,抽象出来工厂接口,将静态工厂中的每个分支判断都抽成单独的工厂类,这也是前文提到过的将if…else分支判断转成单独的类,使代码结构更清晰。

里氏代换原则

       Liskov Substitution Principle,简称LSP。子类型必须能替换掉他们的父类型。换句话说,在软件里面,把使用父类的地方,都换成子类,程序的行为是没有变化的。这是面向对象语言继承的基础,如果没有里氏代换原则,也就谈不上开放-关闭原则了,比如:

动物 animal = new 动物();animal.吃();animal.喝();animal.叫();

       根据里氏代换原则,将动物对象换成狗、牛、骆驼,其它地方不变,是完全可以执行的。

依赖倒置原则

       也叫依赖倒转原则,Dependence Inversion Principle,对抽象进行编程,不要对实现进行编程。面相对象设计将就高内聚、低耦合,高内聚指功能相关性很强的紧密的结合在一起,遵守单一职责原则;耦合是指不同模块或不同层次代码的联系程度,耦合性越低,模块独立性越强,耦合性越高,模块独立性越弱,高耦合的软件,往往修改一处地方,都会牵一发而动全身,这就是设计不当。举个硬件的例子,内存条和主板是通过金手指和主板内存槽连接的,主板定义好插内存条的接口,各个内存条厂商都根据这个标准接口去制作自己的内存条,假如内存条坏了,不能开机了,我不需要去换主板,到中关村买个新的内存条插上即可(话说显卡什么时候能这样做啊,所谓的“独显”都不是真正的独显)。

总结

       工厂方法模式遵循了开放-封闭原则,其实它是将判断工作转移到了客户端调用之处,这样保证了工厂体系的完整性。而简单工厂模式需要在工厂类里来判断选择哪个运算类。

       不知不觉过了午夜了,我对设计模式理解的还很浅显,平时工作用的不算多,现在仅局限于自己学习、试验、总结所获,每学一设计模式就对面向对象的思想理解加深了几分,同时感觉到了自己的渺小与卑微,学到的越多,知道的越多,不知道的越多,自己越渺小。

参考

         《大话设计模式》 程杰著

         维基百科

         百度百科

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏向治洪

Python高效编程技巧

###Python高效编程技巧 如果你发现一个好的程序库,符合你的要求,不要不好意思————大部分的开源项目都欢迎捐赠代码和欢迎提供帮助——即使你不是一个Pyt...

2005
来自专栏知识分享

1-关于单片机通信数据传输(中断发送,大小端,IEEE754浮点型格式,共用体,空闲中断,环形队列)

写这篇文章的目的呢,如题目所言,我承认自己是一个程序猿.....应该说很多很多学单片机的对于...先不说别的了,,无论是学51的还是32的,,,先问一下大家用串...

2485
来自专栏牛客网

贝壳面试流程(技术岗)

首先去了签到 然后到等待 等待区有零食及水供大家食用 到了面试时间 开始一面 一面就是基础 一面过了去三楼等待 如果通知你一会二面 就下去一楼二面(如果是叫到你...

461
来自专栏木宛城主

Thinking In Design Pattern——工厂模式演绎

我始终认为学习设计模式需要怀着一颗敬畏的心去探索,这一系列23种设计模式并不是一蹴而就,都是前人根据自己的经验逐渐演化出来,所以才会形成非常经典的理论。学习设...

2339
来自专栏web前端教室

如何阅读JS源码?读源码有什么好处

这几天在公司接手了一个项目,是之前其它组的,现在要继续完成它。那我要做的第一件事,就是熟悉代码。对,就是看别人写的JS代码。文档嘛,自然是没有的。 之前也有试过...

30210
来自专栏一个会写诗的程序员的博客

Java新手极简指北手册

为什么我先拿“数据结构和算法”说事捏?这玩意是写程序最最基本的东东。不管你使用 Java 还是其它的什么语言,都离不开它。而且这玩意是跨语言的,学好之后不管在哪...

651
来自专栏PPV课数据科学社区

数据结构与算法–关键路径

关键路径与无环加权有向图的最长路径 现在考虑一个这样的问题:你今天事情比较多,要洗衣服、做作业还要烧水洗澡,之后出去找朋友玩。假设洗衣服要20分钟,烧水要30分...

2707
来自专栏Phoenix的Android之旅

工厂模式之抽象工厂

然后挖了个坑…没有说 Abstract Factory,主要是因为这两种模式之间相似度比较高, 一起讲容易混淆,所以就另起一篇来说了。

932
来自专栏Java学习网

想成为优秀程序员必知的要点

想成为优秀程序员必知的要点 1.永远不要重复代码 不惜一切代价也要避免重复代码。如果你有几个不同的地方经常性地要使用某个代码片段,那么可以将它重构成函数。代码...

2097
来自专栏互联网杂技

如何去了解JavaScript引擎的工作原理

1. 什么是JavaScript解析引擎? 简单地说,JavaScript解析引擎就是能够“读懂”JavaScript代码,并准确地给出代码运行结果的一段程序。...

3387

扫码关注云+社区