一、引子
話說十年前,有一個暴發(fā)戶,他家有三輛汽車——Benz奔馳、Bmw寶馬、Audi奧迪,還雇了司機(jī)為他開車。不過,暴發(fā)戶坐車時總是怪怪的:上Benz 車后跟司機(jī)說“開奔馳車!”,坐上Bmw后他說“開寶馬車!”,坐上Audi說“開奧迪車!”。你一定說:這人有病!直接說開車不就行了?!
而當(dāng)把這個暴發(fā)戶的行為放到我們程序設(shè)計中來時,會發(fā)現(xiàn)這是一個普遍存在的現(xiàn)象。幸運(yùn)的是,這種有病的現(xiàn)象在OO(面向?qū)ο螅┱Z言中可以避免了。下面就以 Java語言為基礎(chǔ)來引入我們本文的主題:工廠模式。
二、分類
工廠模式主要是為創(chuàng)建對象提供過渡接口,以便將創(chuàng)建對象的具體過程屏蔽隔離起來,達(dá)到提高靈活性的目的。
工廠模式在《Java與模式》中分為三類:
1)簡單工廠模式(Simple Factory)
2)工廠方法模式(Factory Method)
3)抽象工廠模式(Abstract Factory)
這三種模式從上到下逐步抽象,并且更具一般性。
GOF在《設(shè)計模式》一書中將工廠模式分為兩類:工廠方法模式(Factory Method)與抽象工廠模式(Abstract Factory)。將簡單工廠模式(Simple Factory)看為工廠方法模式的一種特例,兩者歸為一類。
兩者皆可,在本文使用《Java與模式》的分類方法。下面來看看這些工廠模式是怎么來“治病”的。
三、簡單工廠模式
簡單工廠模式又稱靜態(tài)工廠方法模式。重命名上就可以看出這個模式一定很簡單。它存在的目的很簡單:定義一個用于創(chuàng)建對象的接口。
先來看看它的組成:
1) 工廠類角色:這是本模式的核心,含有一定的商業(yè)邏輯和判斷邏輯。在java中它往往由一個具體類實現(xiàn)。
2) 抽象產(chǎn)品角色:它一般是具體產(chǎn)品繼承的父類或者實現(xiàn)的接口。在java中由接口或者抽象類來實現(xiàn)。
3) 具體產(chǎn)品角色:工廠類所創(chuàng)建的對象就是此角色的實例。在java中由一個具體類實現(xiàn)。
那么簡單工廠模式怎么來使用呢?我們就以簡單工廠模式來改造暴發(fā)戶坐車的方式——現(xiàn)在暴發(fā)戶只需要坐在車?yán)飳λ緳C(jī)說句:“開車”就可以了。
將本程序空缺的其他信息填充完整后即可運(yùn)行。如果你將所有的類放在一個文件中,請不要忘記只能有一個類被聲明為public。本程序在jdk1.4 下運(yùn)行通過。
這便是簡單工廠模式了。怎么樣,使用起來很簡單吧?那么它帶來了什么好處呢?
首先,使用了簡單工廠模式后,我們的程序不在“有病”,更加符合現(xiàn)實中的情況;而且客戶端免除了直接創(chuàng)建產(chǎn)品對象的責(zé)任,而僅僅負(fù)責(zé)“消費(fèi)”產(chǎn)品(正如暴發(fā)戶所為)。
下面我們從開閉原則(對擴(kuò)展開放;對修改封閉)上來分析下簡單工廠模式。當(dāng)暴發(fā)戶增加了一輛車的時候,只要符合抽象產(chǎn)品制定的合同,那么只要通知工廠類知道就可以被客戶使用了。所以對產(chǎn)品部分來說,它是符合開閉原則的;但是工廠部分好像不太理想,因為每增加一輛車,都要在工廠類中增加相應(yīng)的業(yè)務(wù)邏輯或者判斷邏輯,這顯然是違背開閉原則的。可想而知對于新產(chǎn)品的加入,工廠類是很被動的。對于這樣的工廠類(在我們的例子中是為司機(jī)師傅),我們稱它為全能類或者上帝類。
我們舉的例子是最簡單的情況,而在實際應(yīng)用中,很可能產(chǎn)品是一個多層次的樹狀結(jié)構(gòu)。由于簡單工廠模式中只有一個工廠類來對應(yīng)這些產(chǎn)品,所以這可能會把我們的上帝累壞了,也累壞了我們這些程序員:(
于是工廠方法模式作為救世主出現(xiàn)了。
話說十年前,有一個暴發(fā)戶,他家有三輛汽車——Benz奔馳、Bmw寶馬、Audi奧迪,還雇了司機(jī)為他開車。不過,暴發(fā)戶坐車時總是怪怪的:上Benz 車后跟司機(jī)說“開奔馳車!”,坐上Bmw后他說“開寶馬車!”,坐上Audi說“開奧迪車!”。你一定說:這人有病!直接說開車不就行了?!
而當(dāng)把這個暴發(fā)戶的行為放到我們程序設(shè)計中來時,會發(fā)現(xiàn)這是一個普遍存在的現(xiàn)象。幸運(yùn)的是,這種有病的現(xiàn)象在OO(面向?qū)ο螅┱Z言中可以避免了。下面就以 Java語言為基礎(chǔ)來引入我們本文的主題:工廠模式。
二、分類
工廠模式主要是為創(chuàng)建對象提供過渡接口,以便將創(chuàng)建對象的具體過程屏蔽隔離起來,達(dá)到提高靈活性的目的。
工廠模式在《Java與模式》中分為三類:
1)簡單工廠模式(Simple Factory)
2)工廠方法模式(Factory Method)
3)抽象工廠模式(Abstract Factory)
這三種模式從上到下逐步抽象,并且更具一般性。
GOF在《設(shè)計模式》一書中將工廠模式分為兩類:工廠方法模式(Factory Method)與抽象工廠模式(Abstract Factory)。將簡單工廠模式(Simple Factory)看為工廠方法模式的一種特例,兩者歸為一類。
兩者皆可,在本文使用《Java與模式》的分類方法。下面來看看這些工廠模式是怎么來“治病”的。
三、簡單工廠模式
簡單工廠模式又稱靜態(tài)工廠方法模式。重命名上就可以看出這個模式一定很簡單。它存在的目的很簡單:定義一個用于創(chuàng)建對象的接口。
先來看看它的組成:
1) 工廠類角色:這是本模式的核心,含有一定的商業(yè)邏輯和判斷邏輯。在java中它往往由一個具體類實現(xiàn)。
2) 抽象產(chǎn)品角色:它一般是具體產(chǎn)品繼承的父類或者實現(xiàn)的接口。在java中由接口或者抽象類來實現(xiàn)。
3) 具體產(chǎn)品角色:工廠類所創(chuàng)建的對象就是此角色的實例。在java中由一個具體類實現(xiàn)。
那么簡單工廠模式怎么來使用呢?我們就以簡單工廠模式來改造暴發(fā)戶坐車的方式——現(xiàn)在暴發(fā)戶只需要坐在車?yán)飳λ緳C(jī)說句:“開車”就可以了。
- //抽象產(chǎn)品角色
- public interface Car{
- public void drive();
- }
- //具體產(chǎn)品角色
- public class Benz implements Car{
- public void drive() {
- System.out.println("Driving Benz ");
- }
- }
- public class Bmw implements Car{
- public void drive() {
- System.out.println("Driving Bmw ");
- }
- }
- ...(奧迪我就不寫了)
- //工廠類角色
- public class Driver{
- //工廠方法.注意 返回類型為抽象產(chǎn)品角色
- public static Car driverCar(String s)throws Exception{
- //判斷邏輯,返回具體的產(chǎn)品角色給Client
- if(s.equalsIgnoreCase("Benz"))
- return new Benz();
- else if(s.equalsIgnoreCase("Bmw"))
- return new Bmw();
- ......
- else throw new Exception();
- ...
- //歡迎暴發(fā)戶出場......
- public class Magnate{
- public static void main(String[] args){
- try{
- //告訴司機(jī)我今天坐奔馳
- Car car = Driver.driverCar("benz");
- //下命令:開車
- car.drive();
- ...
將本程序空缺的其他信息填充完整后即可運(yùn)行。如果你將所有的類放在一個文件中,請不要忘記只能有一個類被聲明為public。本程序在jdk1.4 下運(yùn)行通過。
這便是簡單工廠模式了。怎么樣,使用起來很簡單吧?那么它帶來了什么好處呢?
首先,使用了簡單工廠模式后,我們的程序不在“有病”,更加符合現(xiàn)實中的情況;而且客戶端免除了直接創(chuàng)建產(chǎn)品對象的責(zé)任,而僅僅負(fù)責(zé)“消費(fèi)”產(chǎn)品(正如暴發(fā)戶所為)。
下面我們從開閉原則(對擴(kuò)展開放;對修改封閉)上來分析下簡單工廠模式。當(dāng)暴發(fā)戶增加了一輛車的時候,只要符合抽象產(chǎn)品制定的合同,那么只要通知工廠類知道就可以被客戶使用了。所以對產(chǎn)品部分來說,它是符合開閉原則的;但是工廠部分好像不太理想,因為每增加一輛車,都要在工廠類中增加相應(yīng)的業(yè)務(wù)邏輯或者判斷邏輯,這顯然是違背開閉原則的。可想而知對于新產(chǎn)品的加入,工廠類是很被動的。對于這樣的工廠類(在我們的例子中是為司機(jī)師傅),我們稱它為全能類或者上帝類。
我們舉的例子是最簡單的情況,而在實際應(yīng)用中,很可能產(chǎn)品是一個多層次的樹狀結(jié)構(gòu)。由于簡單工廠模式中只有一個工廠類來對應(yīng)這些產(chǎn)品,所以這可能會把我們的上帝累壞了,也累壞了我們這些程序員:(
于是工廠方法模式作為救世主出現(xiàn)了。
工廠方法模式去掉了簡單工廠模式中工廠方法的靜態(tài)屬性,使得它可以被子類繼承。這樣在簡單工廠模式里集中在工廠方法上的壓力可以由工廠方法模式里不同的工廠子類來分擔(dān)。
你應(yīng)該大致猜出了工廠方法模式的結(jié)構(gòu),來看下它的組成:
1)抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無關(guān)。是具體工廠角色必須實現(xiàn)的接口或者必須繼承的父類。在java中它由抽象類或者接口來實現(xiàn)。
2)具體工廠角色:它含有和具體業(yè)務(wù)邏輯有關(guān)的代碼。由應(yīng)用程序調(diào)用以創(chuàng)建對應(yīng)的具體產(chǎn)品的對象。
3)抽象產(chǎn)品角色:它是具體產(chǎn)品繼承的父類或者是實現(xiàn)的接口。在java中一般有抽象類或者接口來實現(xiàn)。
4)具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對象就是此角色的實例。在java中由具體的類來實現(xiàn)。
工廠方法模式使用繼承自抽象工廠角色的多個子類來代替簡單工廠模式中的“上帝類”。正如上面所說,這樣便分擔(dān)了對象承受的壓力;而且這樣使得結(jié)構(gòu)變得靈活起來——當(dāng)有新的產(chǎn)品(即暴發(fā)戶的汽車)產(chǎn)生時,只要按照抽象產(chǎn)品角色、抽象工廠角色提供的合同來生成,那么就可以被客戶使用,而不必去修改任何已有的代碼。可以看出工廠角色的結(jié)構(gòu)也是符合開閉原則的!
我們還是老規(guī)矩,使用一個完整的例子來看看工廠模式各個角色之間是如何來協(xié)調(diào)的。話說暴發(fā)戶生意越做越大,自己的愛車也越來越多。這可苦了那位司機(jī)師傅了,什么車它都要記得,維護(hù),都要經(jīng)過他來使用!于是暴發(fā)戶同情他說:看你跟我這么多年的份上,以后你不用這么辛苦了,我給你分配幾個人手,你只管管好他們就行了!于是,工廠方法模式的管理出現(xiàn)了。代碼如下:
可以看出工廠方法的加入,使得對象的數(shù)量成倍增長。當(dāng)產(chǎn)品種類非常多時,會出現(xiàn)大量的與之對應(yīng)的工廠對象,這不是我們所希望的。因為如果不能避免這種情況,可以考慮使用簡單工廠模式與工廠方法模式相結(jié)合的方式來減少工廠類:即對于產(chǎn)品樹上類似的種類(一般是樹的葉子中互為兄弟的)使用簡單工廠模式來實現(xiàn)。
五、小結(jié)
工廠方法模式仿佛已經(jīng)很完美的對對象的創(chuàng)建進(jìn)行了包裝,使得客戶程序中僅僅處理抽象產(chǎn)品角色提供的接口。那我們是否一定要在代碼中遍布工廠呢?大可不必。也許在下面情況下你可以考慮使用工廠方法模式:
1)當(dāng)客戶程序不需要知道要使用對象的創(chuàng)建過程。
2)客戶程序使用的對象存在變動的可能,或者根本就不知道使用哪一個具體的對象。
簡單工廠模式與工廠方法模式真正的避免了代碼的改動了?沒有。在簡單工廠模式中,新產(chǎn)品的加入要修改工廠角色中的判斷語句;而在工廠方法模式中,要么將判斷邏輯留在抽象工廠角色中,要么在客戶程序中將具體工廠角色寫死(就象上面的例子一樣)。而且產(chǎn)品對象創(chuàng)建條件的改變必然會引起工廠角色的修改。
面對這種情況,Java的反射機(jī)制與配置文件的巧妙結(jié)合突破了限制——這在Spring中完美的體現(xiàn)了出來。
先來認(rèn)識下什么是產(chǎn)品族: 位于不同產(chǎn)品等級結(jié)構(gòu)中,功能相關(guān)聯(lián)的產(chǎn)品組成的家族。還是讓我們用一個例子來形象地說明一下吧。
回到抽象工廠模式的話題上。
可以說,抽象工廠模式和工廠方法模式的區(qū)別就在于需要創(chuàng)建對象的復(fù)雜程度上。而且抽象工廠模式是三個里面最為抽象、最具一般性的。
抽象工廠模式的用意為:給客戶端提供一個接口,可以創(chuàng)建多個產(chǎn)品族中的產(chǎn)品對象
而且使用抽象工廠模式還要滿足一下條件:
1)系統(tǒng)中有多個產(chǎn)品族,而系統(tǒng)一次只可能消費(fèi)其中一族產(chǎn)品。
2)同屬于同一個產(chǎn)品族的產(chǎn)品以其使用。
來看看抽象工廠模式的各個角色(和工廠方法的如出一轍):
1)抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無關(guān)。是具體工廠角色必須實現(xiàn)的接口或者必須繼承的父類。在java中它由抽象類或者接口來實現(xiàn)。
2)具體工廠角色:它含有和具體業(yè)務(wù)邏輯有關(guān)的代碼。由應(yīng)用程序調(diào)用以創(chuàng)建對應(yīng)的具體產(chǎn)品的對象。在java中它由具體的類來實現(xiàn)。
3)抽象產(chǎn)品角色:它是具體產(chǎn)品繼承的父類或者是實現(xiàn)的接口。在java中一般有抽象類或者接口來實現(xiàn)。
4)具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對象就是此角色的實例。在java中由具體的類來實現(xiàn)。
看過了前兩個模式,對這個模式各個角色之間的協(xié)調(diào)情況應(yīng)該心里有個數(shù)了,我就不舉具體的例子了。只是一定要注意滿足使用抽象工廠模式的條件哦。