`
JAVA天地
  • 浏览: 673709 次
  • 性别: Icon_minigender_1
  • 来自: 太原
文章分类
社区版块
存档分类
最新评论

工厂模式我之见(简单工厂模式 VS 工厂方法模式)(转载)

阅读更多
前言
工大有许多同学是做java的,大家都知道java最大的优点是它的完全OO化和它在多年的发展过程中吸收和总结了许多先进的框架与模式,其中工厂模式就是最常用的模式之一。下面我想将我在学习和实践过程中对工厂模式的认识与了解介绍给大家。由于笔者能力限制,在实践中也没参与过什么大的项目,笔者参与过的项目用到的工厂模式主要是简单工厂模式(Simple Factory)和工厂方法模式(Factory Method),所以笔者在本文主要介绍的是这两种模式。



准备知识
在OO设计领域,我们知道前人总结了不少的经验,许多的经验在现代软件工程过程中已经被认为是原则来遵守。下面笔者摘抄几项下文涉及到的OO原则的定义。

OCP(开闭原则,Open-Closed Principle):一个软件的实体应当对扩展开放,对修改关闭。我的理解是,对于一个已有的软件,如果需要扩展,应当在不需修改已有代码的基础上进行。

DIP(依赖倒转原则,Dependence Inversion Principle):要针对接口编程,不要针对实现编程。我的理解是,对于不同层次的编程,高层次暴露给低层次的应当只是接口,而不是它的具体类。

LoD(迪米特法则,Law of Demeter):只与你直接的朋友通信,而避免和陌生人通信。众所周知类(或模块)之间的通信越少,耦合度就越低,从而更有利于我们对软件的宏观管理。老子论“圣人之治”有相同的思想,《老子》云:“是以圣人之治,虚其心,实其腹,弱其志,常使民无知无欲。”,又云:“小国寡民,邻国相望,鸡犬之声相闻,民至老死,不相往来。”。佩服我们的老祖宗,N千年前就想到了西方N千年后才想到的东西,同时也佩服《java与模式》的作者阎宏,可以用中国传统哲学思想这么生动的说明这一软件设计原则。



简单工厂模式及实例
简单工厂模式又叫静态工厂模式,顾名思义,它是用来实例化目标类的静态类。下面我主要通过一个简单的实例说明简单工厂及其优点。

比如有个国家的运动员协会,他们是负责登记与注册职业运动员的(就好像我们国家的体育总局,呵呵,无论足球篮球还是乒乓球的运动员都必须在这里注册才能拿到我们国家职业运动员牌照)。一家体育俱乐部(比如篮球的广东宏远,足球的深圳健力宝)想获得球员为自己俱乐部效力,就必须通过这个运动员协会。

根据DIP我们可以设计一个“运动员”接口,“足球运动员”和“篮球运动员”(还有其他运动员)都实现“运动员”这个接口。而“运动员协会”就是一个简单工厂类,它负责实例化“运动员”。我们这里的“俱乐部”就是一个客户端(Client),不同的“俱乐部”就是不同的客户端。具体如下图表示:




对于不同的俱乐部对象(无论是八一还是深圳健力宝),他们都是面向“运动员”接口编程,而不用管是“足球运动员”还是“篮球运动员”,也就是说实现了“运动员”接口的具体类“足球运动员”无需暴露给客户端。这也满足了DIP。但具体的俱乐部(比如足球的深圳健力宝)如何确保自己获取的是自己想要的运动员(健力宝俱乐部需要的当然是足球运动员)呢?这就需要“运动员协会”这一工厂类了。俱乐部通过调用“运动员协会”的具体方法,返回不同的实例。这同时也满足了LoD,也就是“深圳健力宝足球俱乐部”对象不直接与“足球运动员:李毅”对象通信,而是通过他们共同的“朋友”——“国家体育总局”通信。

下面给出各个类的程序,会有助于读者更好的了解笔者之前的介绍。

Code: [Copy to clipboard]
运动员.java
public interface 运动员 {
public void 跑();
public void 跳();
}

足球运动员.java
public class 足球运动员 implements 运动员 {

public void 跑(){
//跑啊跑
}

public void 跳(){
//跳啊跳
}
}

篮球运动员.java
public class 篮球运动员 implements 运动员 {

public void 跑(){
//do nothing
}

public void 跳(){
//do nothing
}
}

体育协会.java
public class 体育协会 {

public static 运动员 注册足球运动员(){
return new 足球运动员();
}

public static 运动员 注册篮球运动员(){
return new 篮球运动员();
}

}

俱乐部.java
public class 俱乐部 {
private 运动员 守门员;
private 运动员 后卫;
private 运动员 前锋;

public void test() {
this.前锋 = 体育协会.注册足球运动员();
this.后卫 = 体育协会.注册足球运动员();
this.守门员 = 体育协会.注册足球运动员();

守门员.跑();
后卫.跳();
}
}

以上就是简单工厂模式的一个简单实例,读者应该想象不用接口不用工厂而把具体类暴露给客户端的那种混乱情形吧(就好像没了体育总局,各个俱乐部在市场上自己胡乱的寻找仔细需要的运动员),简单工厂就解决了这种混乱。

我们用OCP看看简单工厂,会发现如果要对系统进行扩展的话治需要增加实现产品接口的产品类(上例表现为“足球运动员”,“篮球运动员”类,比如要增加个“乒乓球运动员”类),而无需对原有的产品类进行修改。这咋一看好像满足OCP,但是实际上还是需要修改代码的——对,就是修改工厂类。上例中如果增加“乒乓球运动员”产品类,就必须相应的修改“体育协会”工厂类,增加个“注册乒乓球运动员”方法。所以可以看出,简单工厂模式是不满足OCP的。



工厂方法模式及其实例
谈了简单工厂模式,下面继续谈谈工厂方法模式。前一节的最末点明了简单工厂模式最大的缺点——不完全满足OCP。为了解决这一缺点,设计师们提出了工厂方法模式。工厂方法模式和简单工厂模式最大的不同在于,简单工厂模式只有一个(对于一个项目或者一个独立模块而言)工厂类,而工厂方法模式有一组实现了相同接口的工厂类。下面我们通过修改上一节的实例来介绍工厂方法模式。

我们在不改变产品类(“足球运动员”类和“篮球运动员”类)的情况下,修改下工厂类的结构,如下图所示:




相关代码如下:

Code: [Copy to clipboard]
运动员.java
public interface 运动员 {
public void 跑();
public void 跳();
}

足球运动员.java
public class 足球运动员 implements 运动员 {

public void 跑(){
//跑啊跑
}

public void 跳(){
//跳啊跳
}
}

篮球运动员.java
public class 篮球运动员 implements 运动员 {

public void 跑(){
//do nothing
}

public void 跳(){
//do nothing
}
}

体育协会.java
public interface 体育协会 {
public 运动员 注册();
}

足球协会.java
public class 足球协会 implements 体育协会 {
public 运动员 注册(){
return new 足球运动员();
}
}

篮球协会.java
public class 篮球协会 implements 体育协会 {
public 运动员 注册(){
return new 篮球运动员();
}
}

俱乐部.java
public class 俱乐部 {
private 运动员 守门员;
private 运动员 后卫;
private 运动员 前锋;

public void test() {
体育协会 中国足协 = new 足球协会();

this.前锋 = 中国足协.注册();
this.后卫 = 中国足协.注册();

守门员.跑();
后卫.跳();
}
}


很明显可以看到,“体育协会”工厂类变成了“体育协会”接口,而实现此接口的分别是“足球协会”“篮球协会”等等具体的工厂类。

这样做有什么好处呢?很明显,这样做就完全OCP了。如果需要再加入(或扩展)产品类(比如加多个“乒乓球运动员”)的话就不再需要修改工厂类了,而只需相应的再添加一个实现了工厂接口(“体育协会”接口)的具体工厂类。



简单工厂模式与工厂方法模式大PK
从以上对两种模式的介绍可以了解到,工厂方法模式是为了克服简单工厂模式的缺点(主要是为了满足OCP)而设计出来的。但是,工厂方法模式就一定比简单工厂模式好呢?笔者的答案是不一定。下面笔者将详细比较两种模式。

1. 结构复杂度
从这个角度比较,显然简单工厂模式要占优。简单工厂模式只需一个工厂类,而工厂方法模式的工厂类随着产品类个数增加而增加,这无疑会使类的个数越来越多,从而增加了结构的复杂程度。

2.代码复杂度
代码复杂度和结构复杂度是一对矛盾,既然简单工厂模式在结构方面相对简洁,那么它在代码方面肯定是比工厂方法模式复杂的了。简单工厂模式的工厂类随着产品类的增加需要增加很多方法(或代码),而工厂方法模式每个具体工厂类只完成单一任务,代码简洁。

3.客户端编程难度
工厂方法模式虽然在工厂类结构中引入了接口从而满足了OCP,但是在客户端编码中需要对工厂类进行实例化。而简单工厂模式的工厂类是个静态类,在客户端无需实例化,这无疑是个吸引人的优点。

4.管理上的难度
这是个关键的问题。
我们先谈扩展。众所周知,工厂方法模式完全满足OCP,即它有非常良好的扩展性。那是否就说明了简单工厂模式就没有扩展性呢?答案是否定的。简单工厂模式同样具备良好的扩展性——扩展的时候仅需要修改少量的代码(修改工厂类的代码)就可以满足扩展性的要求了。尽管这没有完全满足OCP,但笔者认为不需要太拘泥于设计理论,要知道,sun提供的java官方工具包中也有想到多没有满足OCP的例子啊(java.util.Calendar这个抽象类就不满足OCP,具体原因大家可以分析下)。
然后我们从维护性的角度分析下。假如某个具体产品类需要进行一定的修改,很可能需要修改对应的工厂类。当同时需要修改多个产品类的时候,对工厂类的修改会变得相当麻烦(对号入座已经是个问题了)。反而简单工厂没有这些麻烦,当多个产品类需要修改是,简单工厂模式仍然仅仅需要修改唯一的工厂类(无论怎样都能改到满足要求吧?大不了把这个类重写)。

由以上的分析,笔者认为简单工厂模式更好用更方便些。当然这只是笔者的个人看法而已,毕竟公认的,工厂方法模式比简单工厂模式更“先进”。但有时过于先进的东西未必适合自己,这个见仁见智吧。


写在最后
本文仅讨论了两个常见的工厂模式,笔者很主观的分析了各自的优缺点,这必定会引起许多异议。如果对笔者观点有意见的,很欢迎跟贴批评指出。

其实究竟哪种模式更好,可能你会回答“都不好”,呵呵,别忘了,我们还有更“先进”的“抽象工厂模式”。。。

最后交代下版权信息。本文许多定义性的内容引用自《java与模式》(电子工业出版社,阎宏),这是本好书,有时间有机会希望大家可以好好看看,我在此书中是获益良多。另外笔者在写本文过程中还参考了《UML基础、案例与应用》(人民邮电出版社,Joseph Schmuller著,李虎、赵龙刚译),除此外其余内容均为原创,转贴请注明“工大后院版权所有”。

分享到:
评论

相关推荐

    抽象工厂模式 源代码

    此资源出自下面的作者,我只是转载,非常实用的设计方法,如果您想成位出色的设计师,那就再复习复习吧!如果您想成为软件设计师,通过学习,您将会站另一个高度看待软件设计. 原始地址:...

    二十三种设计模式【PDF版】

    使用工厂模式就象使用 new 一样频繁. 设计模式之 Builder 汽车由车轮 方向盘 发动机很多部件组成,同时,将这些部件组装成汽车也是一件复杂的工作,Builder 模式就是将这两 种情况分开进行。 设计模式之 ...

    android设计模式详解转载

    Android中的LayoutInflater就是一个典型的工厂模式应用,它负责将XML布局文件转换为视图对象。 3. 构造器模式(Builder模式):将复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。在Android...

    Java23种设计模式

    2. 抽象工厂模式(Abstract Factory):提供一个接口,用于创建相关或依赖对象的家族,而不需要明确指定具体类。 适用场景:系统中有多个产品族,而每次只使用其中的一个产品族;系统提供一个产品类的库,所有的产品...

    Java程序员面试的试题集(1_122)帮助初学者的技术问题(转载)

    ### Java程序员面试试题详解 #### CORBA的理解与应用 CORBA,全称为Common Object...在J2EE开发中,工厂模式常用于创建各种类型的组件,如数据库连接、业务对象等,简化了系统架构,增强了代码的可维护性和可测试性。

    悠索科技高校教务管理系统(转载)

    5. **设计模式**:系统可能应用了诸如工厂模式、单例模式、观察者模式等设计模式,以实现良好的代码结构和可扩展性。 6. **权限控制与安全**:考虑到教务管理系统涉及敏感信息,系统可能包含了用户角色、权限分配等...

    [转载]QQ示例源码(供学习C++网络编程参考)

    9. **设计模式**:在实现大型网络应用时,设计模式如工厂模式、观察者模式和状态机模式等可以帮助构建可扩展和可维护的代码。 10. **网络库**:为了简化网络编程,有许多优秀的C++库可供使用,如Boost.Asio、Poco库...

    [转载]+[C#]+加强型音乐播放器+代码类

    良好的设计模式,如单例模式(用于控制播放器实例的数量)和工厂模式(用于创建音频对象)也可能被应用。 最后,考虑到“很实用”,这个音乐播放器可能还包括了一些高级特性,比如音轨转换、歌词同步显示、网络流...

    Java面试题

    8. **设计模式**:常见的设计模式如单例模式、工厂模式、建造者模式、观察者模式、适配器模式等,以及在实际开发中的应用。 9. **JVM优化**:了解JVM内存模型(堆、栈、方法区等),垃圾回收机制(GC),以及如何...

    转载:软件开发者面试百问

    - 设计模式:如工厂模式、观察者模式等,它们在解决常见问题时提供了一种标准解决方案。 - 无状态业务层:理解其含义并讨论长事务的处理方式。 - 架构图:如用例图、类图、序列图等,用于表示系统组件及其相互...

    java编程思想习题及答案

    10. **设计模式**:习题可能涉及到一些常见的设计模式,如单例、工厂、观察者、装饰器等,设计模式是解决常见编程问题的通用解决方案。 通过解答这些习题,学习者不仅能巩固Java语言的基础,还能提高解决实际问题的...

    小小图片爬虫

    11. **代码结构与设计模式**:良好的代码组织和设计模式(如工厂模式、观察者模式)可以使项目更易于维护和扩展。 通过学习和实践这个【小小图片爬虫】项目,开发者不仅可以掌握HTTP请求的基本操作,还能了解到如何...

    抽奖软件java

    8. **设计模式**:良好的软件设计通常会运用设计模式,比如单例模式用于控制抽奖逻辑的实例化,工厂模式用于创建GUI组件,观察者模式用于更新界面状态等。 9. **程序测试**:代码已经测试并确认无误,这意味着...

    DWM1000中文手册

    由于其精确的定位功能、高速数据传输能力以及低功耗特性,DWM1000成为智能楼宇、工厂自动化和各种物联网应用中的理想选择。 文档的最后也提醒用户,虽然DecaWave公司尽力确保文档中的信息是准确的,但所提供的信息...

    Java工程师新人入门书籍推荐

    - 《漫谈设计模式:从面向对象开始》:介绍了设计模式,对于软件设计至关重要,尤其是单例、工厂和代理模式。 - 《Spring 3.0就这么简单》:适合初学者快速掌握Spring框架的入门书籍。 - 《Java并发编程实战》:...

    Java面试资料大集合

    Java是世界上最流行的编程语言之一,尤其在企业级应用开发领域占据主导地位。为了在激烈的求职竞争中脱颖而出,理解和掌握Java的面试知识点至关重要。本资料集合提供了丰富的Java面试资源,涵盖了从基础到高级的各种...

    日本贴片机操作手册,pdf文档

    - **注意事项**:在产品移设到海外工厂或转卖时,必须事先与富士机械制造株式会社或代理商联系,否则可能无法获得备用部件或维修保养服务。 综合以上内容,手册涵盖了高速复合型贴片机XPF从安装、使用、设定、到...

    关于人工智能的随笔-谈人工智能时代的学习.doc

    当被问及对互联网未来的看法时,施密特答道:"我的答案非常简单:互 联网将会消失。"影响最深远的技术是那些消失的技术。它们将自己融入日常之中,以至 我们无可分辨。比方,谷歌已经开发了智能隐形眼镜,能够测量...

    java事例集合1

    9. **设计模式**:可能包含一些常见设计模式的实现,如单例、工厂、观察者等。 10. **异常与日志记录**:如何有效地捕获和记录程序中的异常信息。 11. **反射机制**:用于在运行时动态获取类的信息和操控对象。 ...

    80744924NCGuide(C)-01.pdf

    **发那科(FANUC)**是一家全球领先的自动化技术提供商,专注于数控系统、机器人及工厂自动化系统的研发与制造。本参考资料主要介绍了发那科机床联网的相关技术和应用。 ### 全球分支机构 1. **日本总部** - 地址...

Global site tag (gtag.js) - Google Analytics