java工厂模式 |
|
一、引子
二、简介
工厂模式主要是为创建对象提供了接口。工厂模式按照《Java与模式》中的提法分为三类: 1. 简单工厂模式(Simple Factory) 2. 工厂方法模式(Factory Method) 3. 抽象工厂模式(Abstract Factory) 这三种模式从上到下逐步抽象,并且更具一般性。还有一种分类法,就是将简单工厂模式看为工厂方法模式的一种特例,两个归为一类。下面是使用工厂模式的两种情况: 1.在编码时不能预见需要创建哪种类的实例。 2.系统不应依赖于产品类实例如何被创建、组合和表达的细节
三、简单工厂模式
顾名思义,这个模式本身很简单,而且使用在业务较简单的情况下。 它由三种角色组成(关系见下面的类图): 1、工厂类角色:这是本模式的核心,含有一定的商业逻辑和判断逻辑。在java中它往往由一个具体类实现。
2、抽象产品角色:它一般是具体产品继承的父类或者实现的接口。在java中由接口或者抽象类来实现。
3、具体产品角色:工厂类所创建的对象就是此角色的实例。在java中由一个具体类实现。
650) this.width=650;" style="zoom: 60%; cursor: pointer" alt="" width="500" height="209" resized="true" οnclick="javascript:window.open("http://blog.51cto.com/viewpic.php?refimg=" + this.src);" src="http://blog.csai.cn/UploadFiles/2009-2/26825231.jpg" /> 那么简单工厂模式怎么用呢?我来举个例子吧,我想这个比讲一大段理论上的文字描述要容易理解的多!下面就来给那个暴发户治病: P 在使用了简单工厂模式后,现在暴发户只需要坐在车里对司机说句:"开车"就可以了。来看看怎么实现的: //抽象产品角色 public interface Car{ public void drive(); } //具体产品角色 public class Bmw implements Car{ //工厂类角色 //工厂方法 //判断逻辑,返回具体的产品角色给Client ......
//欢迎暴发户出场......
public class Magnate{ public static void main(String[] args){ try{ //告诉司机我今天坐奔驰 Car car = Driver.driverCar("benz"); //下命令:开车 car.drive(); 。。。 如果将所有的类放在一个文件中,请不要忘记只能有一个类被声明为public。 程序中类之间的关系如下:
650) this.width=650;" style="zoom: 110%; cursor: pointer" alt="" width="500" height="205" resized="true" οnclick="javascript:window.open("http://blog.51cto.com/viewpic.php?refimg=" + this.src);" src="http://blog.csai.cn/UploadFiles/2009-2/26294976.jpg" />
这便是简单工厂模式了。下面是其好处:
首先,使用了简单工厂模式后,我们的程序不在"有病",更加符合现实中的情况;而且客户端免除了直接创建产品对象的责任,而仅仅负责"消费"产品(正如暴发户所为)。
下面我们从开闭原则上来分析下简单工厂模式。当暴发户增加了一辆车的时候,只要符合抽象产品制定的合同,那么只要通知工厂类知道就可以被客户使用了。那么对于产品部分来说,它是符合开闭原则的--对扩展开放、对修改关闭;但是工厂部分好像不太理想,因为每增加一辆车,都要在工厂类中增加相应的商业逻辑和判断逻辑,这显自然是违背开闭原则的。 对于这样的工厂类(在我们的例子中是为司机师傅),我们称它为全能类或者上帝类。 我们举的例子是最简单的情况,而在实际应用中,很可能产品是一个多层次的树状结构。由于简单工厂模式中只有一个工厂类来对应这些产品,所以这可能会把我们的上帝类坏了,进而累坏了我们可爱的程序员:( 正如我前面提到的简单工厂模式适用于业务将简单的情况下。而对于复杂的业务环境可能不太适应阿。这就应该由工厂方法模式来出场了!! 四、工厂方法模式 650) this.width=650;" style="zoom: 110%; cursor: pointer" alt="" width="500" height="211" resized="true" οnclick="javascript:window.open("http://blog.51cto.com/viewpic.php?refimg=" + this.src);" src="http://blog.csai.cn/UploadFiles/2009-2/26950544.jpg" />
我们还是老规矩使用一个完整的例子来看看工厂模式各个角色之间是如何来协调的。话说暴发户生意越做越大,自己的爱车也越来越多。这可苦了那位司机师傅了,什么车它都要记得,维护,都要经过他来使用!于是暴发户同情他说:看你跟我这么多年的份上,以后你不用这么辛苦了,我给你分配几个人手,你只管管好他们就行了!于是,工厂方法模式的管理出现了。代码如下: Car car = driver.driverCar(); 五、小结 六、抽象工厂模式 图中的BmwCar和BenzCar就是两个产品树(产品层次结构);而如图所示的BenzSportsCar和BmwSportsCar就是一个产品族。他们都可以放到跑车家族中,因此功能有所关联。同理BmwBussinessCar和BenzSportsCar也是一个产品族。 650) this.width=650;" style="zoom: 30%; cursor: pointer" alt="" width="500" height="309" resized="true" οnclick="javascript:window.open("http://blog.51cto.com/viewpic.php?refimg=" + this.src);" src="http://blog.csai.cn/UploadFiles/2009-2/26362899.jpg" /> 看过了前两个模式,对这个模式各个角色之间的协调情况应该心里有个数了,我就不举具体的例子了。只是一定要注意满足使用抽象工厂模式的条件哦,不然即使存在了多个产品树,也存在产品族,但是不能使用的。 |