论坛首页 Java企业应用论坛

关于Factory, Abstract Factory, Factory Method, 和Builder...

浏览 17222 次
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
作者 正文
   发表时间:2005-11-19   最后修改:2009-12-14
选这4个模式在一起讨论首先是因为她们的功能比较类似,都是用来制造对象实例的,从归类来说,他们都属于制造类模式(creational patterns),其次她们们在工作中比较常用,由于功能太过相似,往往导致在某个实际问题上让人举棋不定,似乎选哪个都可以解决问题,可是真的选择某个模式后,又会发现不是完全合适。今天我们就来讨论讨论在什么情况下选择什么模式来解决问题比较合适。

考虑到不是所有的朋友都对以上提到的4种模式都了如指掌,在开始讨论之前先简单的介绍一下这4个模式,对这4种模式熟悉的朋友也可以顺便回顾一下。请注意,这里我只会做简短的介绍,如果要比较深入地了解她们的话,还请去看模式相关的书籍,比较浅显易懂的我推荐 “Head First Design Patterns”。如果要在工作中反复参考的我推荐 “Applied Java Patterns”。那本经典的GOF Design Patterns由于写的时间比较早,举的例子不太适合现在的软件开发,我个人认为初学者或者是没有太多时间的朋友没有必要去读那本书,尽管她的确是经典。

Factory Pattern

中文叫工厂模式, 这个是我们在面向对象编程中最常用的模式之一了,她的主要功能就是制造对象,也正是这个原因才叫她为工厂模式,工厂干什么?生产产品。通常情况而言某个工厂所生产的产品总是一个系列的不同种类,大体上相同,细节上有差异。

图1 展示了一个非常简单的例子,通常CarFactory类会提供下列方法来生产车辆, 如代码片断1所示:

public class CarFactory {

       public Car createBusinessCar() {….}

       public Car createSportCar() {…}

}

代码片断 1



图1 Factory Pattern

Abstract Factory Pattern

中文叫抽象工厂模式,顾名思义,就是把工厂模式在进一步抽象化,进一步细化。我们继续沿用上面的例子,不过这次增加产品的种类,如图2所示。由于我们增加了一层分类,当我们要生产某种车的时候就需要询问要哪种车, 商用还是跑车?不然的话就要增加create方法,如果分类多了,create方法就会成几何数量增长。

public class CarFactory {

       public Car createBusinessBMWCar() {….}

       public Car createBusinessBenzCar() {…}

       public Car createSportBMWCar() {….}

       public Car createSportBenzCar() {…}

}

代码片断 2



图2 bad Factory Pattern

从面向对象的角度来讲,不是很好的解决方法,需要进一步抽象。结果就是把CarFactory进行抽象,然后针对不同品牌的车产生不同的工厂。 如图3



图3 Abstract Factory Pattern

这个时候每个工厂类仍然只需要2个方法,如代码片断1所示,用户会在不同需要的情况下得到不同的工厂对象,进而生产出需要的Car。不论这里工厂类如何实现,客户端的代码可以始终不变,类似这样:getCarFactory().createBusinessCar()。

以上两个模式多用于:

1 。 客户端需要依赖制造对象的细节。
2. 一系列类似的对象需要被制造出来。
3.      同种对象不同实体需要在不同的地点不同是时间被制造出来。

Factory Method Pattern

中文叫工厂方法模式, 通常是用于当某个类的功能主要是针对生产出来的对象提供一些相关服务,而且这些对象有共同的接口,至于对象具体是什么,留给她的子类来决定。这样解释起来比较晦涩难懂,我们还是沿用生产汽车的例子,不过不再用工厂,而是改用销售部门,因为销售部门主要为生产出来的车提供相关服务的,比方说,折扣,售后服务等等。代码如代码片断3 所示, 这里CarStore是个抽象类,如何生产需要的车将由子类来进一步实现, 比方说BMWStore就会生产出BMW。图4展示了相关的UML图, 总体看起来图4和图3非常相似, 但是从实现原理和实际应用上来说两者是完全不同的, 图3 Abstract Factory是针对Interface的多种不同的实现,具体使用的

时候是“组和”,用英语术语表达就是“composition”。 而图4 Factory Method是子类对抽象父类继承,进而实现父类的抽象方法, 具体使用的时候是”继承”, 用英语术语表达就是”inheritance”。 也就是很多OO书里说谈到的“is a? or has a?”的问题, composition指的就是has a, 而inheritance指的就是is a。

public abstract class CarStore {

       // Factory method

       protected abstract Car createBusinessCar();

       protected abstract Car createSportCar();

     

public void sell() {

  createBusinessCar();

  discount(…);

....

}

       public double discount(Customer customer) {….}

       public void service() {…}

}

代码片断 3

这个模式多半会和另一个模式Template Method一起使用,事实上代码片断3中方法sell() 就是template Method。具体关于template method pattern的详细讲解已不属于本文的讨论主题 ,感兴趣的朋友请看相关书籍。



图4 Factory Method

Builder Pattern

中文叫生成器模式,我觉得叫构造模式更贴切,通常用于提供尽可能简单的接口来生成比较复杂的对象,这个复杂的对象通常包含有很多其他的对象。有些人认为Builder和Factory Method很相似,有时候是可以互换的,这样理解是错误的。这两个模式的侧重点是不同的,Builder侧重的是构造复杂对象,而且经常是使用到composite模式; Factory Method侧重的是生产各种各样不同的对象,这些对象通常都是比较简单的,一般而言,Factory Method不会和composite联合使用。总的来说这两个模式是相辅相成的,决不能视为同类。具体原因我会在讨论中加以阐述。图5 展示了Builder概念图。



图5 Builder Pattern

考虑到使用composite模式,我们使用另外一个例子,做饭,比较简单,常见,人人都做过,解释起来应该比较好懂。我们先定义好“食品”是对所有吃的物品的概括,做好的炸酱面是食品,煮好的面条是食品,做好的酱是食品,做酱用的料也是食品。当我们去餐馆叫吃炸酱面时,我们只希望要一碗做好的炸酱面, 我们可不会希望对小二说煮面,切黄瓜丝,打鸡蛋,剁肉末,熬酱,等等。我们只希望说:“小二,来碗炸酱面。” 这时候就需要用到builder,  builder会提供一个简单的方法createNoodleWithSource() — 做炸酱面,当顾客点吃炸酱面后,builder就会在内部制造一系列的食品,比方说,煮面,切黄瓜丝,打鸡蛋,剁肉末,熬酱,等等。而这些细节,作为顾客是不需要费心的。

讨论

4种模式都已经简要的介绍完了,到这里大家应该对她们有了大概地了解。从根本上来说,每一个模式都不难理解,真正困难的是选择,在什么情况下用哪个模式才是正确的? 每一个初学者碰到了实际的问题都会产生疑问,好像用那个模式都可以解决问题。我对以上4种模式进行了一番对比,并结合工作中的实际问题进行了思考,并且和同事们展开了讨论(这真是人生一大快事啊!)。我的建议是“xxx教导我们要用发展的眼光来看待这个问题“ 

一般来说,在软件开发过程中,总是把大的问题进行细化,然后从小处入手,逐渐发展壮大。也就是说最开始的时候遇到的问题往往很小,一个类就可以解决了,对于只需要生成简单对象的情况,使用Factory模式;对于需要生成复杂对象的情况,用builder模式,只需要写一个类,借口拉,抽象类拉,完全不需要!目前不需要担心松紧耦合的问题。进一步优化的话,可以写两个类,builder负责生成复杂的对象,而将所有关于生成简单对象的工作转交给Factory,用英语术语来表达的话叫“delegate”, 对应的模式叫Business Delegate(core J2EE Patterns)。

当系统发展壮大后, 需要生成很多种类的复杂对象,导致了需要很多builder来负责生成她们,这个时候就需要做两件事,第一是使用Abstract Factory模式,负责生成相对应的builder类,进而对各种复杂的对象生成进行系统的管理。第二是为各种类别的builder和Factory编写接口,然后利用refactoring将使用类的地方尽可能的替换为使用借口,也就是面向对象编程基本原理之一的“尽量针对接口编程,而不是针对实现类”。如果希望优化系统,减少对象实体的生成,可以视情况而定,采用Singleton模式。

请注意到目前为止,我们没有用到继承,这也是符合面向对象编程的原理的—组合优先于继承(composition, not inheritance)。但是当系统再进一步发展长大后,她已经不能再称为模块,而是成长为框架(framework)时,我们就要考虑适当的使用Factory Method模式了。基本的功能将在父类中加以实现,而具体的对象生成交由子类来完成。如果结合其他模式来理解的话,就是子类中生成对象的工作将转交给其他3中模式类们来完成。如果希望在运行状态(runtime)时动态切换相关的Builder或Factory的话,可以采用Strategy模式。不过,就像很多介绍面向对象编程的书籍建议的那样,不到万不得已不要使用继承,也就是说,切忌在系统还非常简单的情况下就贸然使用Factory Method。



图6 Big picture

到此为止所有4种模式都用上了,整个系统有如下优点:

   1. 健壮,易于扩展,易于维护。
   2. 针对接口,系统内部松散,解耦合。
   3. 基本上避免了代码重复。

这样构件的系统里,我们用到了下列模式:

1.      Factory Pattern

2.      Abstract Factory Pattern

3.      Builder Pattern

4.      Singleton Pattern

5.      Template Method Pattern

6.      Composite Pattern

7.      Strategy Pattern

8.      Business Delegate Pattern

请关注我的Blog
http://spaces.msn.com/members/polygoncell/
http://blog.csdn.net/schnell/
   发表时间:2005-11-21  
楼主对创造类模式颇有研究,我这里有个问题,不知能否讨论一下。
虽然很早以前就在设计模式中学习过这些创造模式,但是我感觉真实的工作中很少有需要使用这些有复杂结构的工厂。
希望楼主可以说一下自己在哪些真实的项目情景中使用过类似abstract factory这些东西。
0 请登录后投票
   发表时间:2005-11-21  
98个人来看,只有老兄一个人回我的贴,我先拜谢了,呵呵!

不知道楼上有没有开发过rich client, 在GUI开发方面就会用到Abstract Factory.
0 请登录后投票
   发表时间:2005-11-21  
有了IOC框架,工厂使用的机会少很多,所以。。。。。。
0 请登录后投票
   发表时间:2005-11-22  
polygoncell兄,不要客气,能与你讨论设计模式我也很高兴。论坛里讨论设计模式的人很少,可能是因为设计模式的层次比较“低”吧。这是题外话。

我做过rich client的GUI,有些自己开发的控件构造器确实用了些工厂,或者Builder(我们用的Builder和你文章中写的Builder不太一样,不过这个没关系)。
但是说实话,GUI的东西很简单,一般不会用到复杂结构的工厂。我们都是尽量将工厂简化再简化,以方便编码人员使用。
所以类似抽象工厂这些有复杂结构的工厂我们从来不用,就算迫不得已要用,也是先做抽象工厂,但最后都包装成简单工厂给编码人员使用。所以我并不认为GUI构建是一个迫切需要复杂工厂模式的良好范例。

我在其他领域使用工厂的情形包括
1. 与外部系统交互的模块,在这里工厂主要用来生成Command以及从Command生成外部系统可识别的数据流,一般来说只需要一个简单工厂就够了,但是内部的算法可能很复杂(主要是一些语法规则分析或者类似的算法)。

2. 没有了,呵呵。

所以你也能看到我的困惑,工厂是个好东西,但是就是用不上。
另外partech兄,我也确实觉得ioc框架能解决一部分工厂的问题,但不是全部问题都可以用ioc框架解决,比如那种需要根据传入参数动态生成产品的工厂,据我所知Spring解决这种问题的方案还是工厂,只不过这个工厂是通过BeanFactory创建并得到的——仍然是工厂。我就是希望能通过讨论明确一下哪里真的迫切需要工厂这种模式。
有点钻牛角尖了,呵呵。
0 请登录后投票
   发表时间:2005-11-22  
模式这种东西确实可以玩出不少花样,只是没有现成的IOC框架来得实在。
简单实用才是真理,如果随着潜在需求的不断发掘,系统使用的模式也变来变去,那倒还不如不用的好。
0 请登录后投票
   发表时间:2005-11-22  
另外,我对大部分设计模式的感觉都类似,就是:
刚接触的时候觉得这模式真是才华横溢的想法,但在实践中总是感觉“就是那么回事”。真正对自己的真实情况有益的模式都是通过融会贯通已有的模式并对之加以变化得到的。
“模式只是一个起点”,这话好像是Martin Fowler说的,实在是有道理。我们的眼光都太短浅了,只看到模式本身。
0 请登录后投票
   发表时间:2005-11-22  
这些模式没有什么好讨论的其实,factory对应着一个产品的族系,而abstract factory则对应可能多个的产品族系,builder封装了一个产生式的过程,长篇泛论这样的模式未免有些学院派,模式的精髓就在于保持接口的干净和尽量少的依赖性以及hollywood principle,还有一点很重要,模式的风格也很重要,无论是高层抽象和低层抽象,整个系统保持概念的一致性也很重要,不过下次讨论模式,希望有系统实例结合论述就好了,另外有了spring ioc之类的东西,根本就可以忘记factory之类的创建型模式了,还有如果有了aop的帮助,还可以舍弃一些模式的赘论
0 请登录后投票
   发表时间:2005-11-22  
感觉建造型模式之重点应该放在哪些些类应该使用哪些类不应该使用上。
简单来说类在一个系统中主要有两类表现形式:
不变类和变类
对于变类应该使用建造型模式,而对于不变类就不适合采用建造型模式。
所以个人感觉对于建造型模式的使用本质应该是对系统中类是变还是不变的一个分析结果。。。。
0 请登录后投票
   发表时间:2005-11-22  
最近做一个新项目,在项目中更多的使用了几个factory。和大家交流一下感受。
为什么有factory,据《Java设计模式》一书说,Java的JVM在生成对象时不符合“开闭”原则和依赖倒置原则,而人们又想实现:只在必须的时候才创建对象。因些才有创建模式的出现,以达到推迟对象的创建。
对于书上讲的自有他的道理,在项目中我有不同的看法:
1.创建模式应该只创建不变对象(即没有状态变化的对象),比如Dao/Bo/数据库连接对象等。
2.因为创建对象的花消大,所以对推迟创建出的对象的进行缓存(注意并发的问题)
3.提供给程序员使用时要尽可能的简单,并清楚的告诉他们使用的“流程",降低难度。我们在项目中更多的是使用simple Factory和单实例模式。
0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics