日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

工厂方法模式与抽象工厂模式

發(fā)布時(shí)間:2024/4/15 编程问答 36 豆豆
生活随笔 收集整理的這篇文章主要介紹了 工厂方法模式与抽象工厂模式 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
本文是轉(zhuǎn)的
一、引子
? ? ? ?話說(shuō)十年前,有一個(gè)暴發(fā)戶,他家有三輛汽車——Benz奔馳、Bmw寶馬、Audi奧迪,還雇了司機(jī)為他開車。不過(guò),暴發(fā)戶坐車時(shí)總是怪怪的:上Benz車后跟司機(jī)說(shuō)“開奔馳車!”,坐上Bmw后他說(shuō)“開寶馬車!”,坐上Audi說(shuō)“開奧迪車!”。你一定說(shuō):這人有病!直接說(shuō)開車不就行了?!
? ? ? ?而當(dāng)把這個(gè)暴發(fā)戶的行為放到我們程序設(shè)計(jì)中來(lái)時(shí),會(huì)發(fā)現(xiàn)這是一個(gè)普遍存在的現(xiàn)象。幸運(yùn)的是,這種有病的現(xiàn)象在OO(面向?qū)ο?#xff09;語(yǔ)言中可以避免了。下面就以Java語(yǔ)言為基礎(chǔ)來(lái)引入我們本文的主題:工廠模式。

二、分類
? ? ? 工廠模式主要是為創(chuàng)建對(duì)象提供過(guò)渡接口,以便將創(chuàng)建對(duì)象的具體過(guò)程屏蔽隔離起來(lái),達(dá)到提高靈活性的目的。
工廠模式在《Java與模式》中分為三類:
? ? ? ?1)簡(jiǎn)單工廠模式(Simple Factory)
? ? ? ?2)工廠方法模式(Factory Method)
? ? ? ?3)抽象工廠模式(Abstract Factory)
? ? ? ?這三種模式從上到下逐步抽象,并且更具一般性。
? ? ? ?GOF在《設(shè)計(jì)模式》一書中將工廠模式分為兩類:工廠方法模式(Factory Method)與抽象工廠模式(Abstract Factory)。將簡(jiǎn)單工廠模式(Simple Factory)看為工廠方法模式的一種特例,兩者歸為一類。兩者皆可,在本文使用《Java與模式》的分類方法。下面來(lái)看看這些工廠模式是怎么來(lái)“治病”的。
?
三、簡(jiǎn)單工廠模式
?
? ? ? ?簡(jiǎn)單工廠模式又稱靜態(tài)工廠方法模式。重命名上就可以看出這個(gè)模式一定很簡(jiǎn)單。它存在的目的很簡(jiǎn)單:定義一個(gè)用于創(chuàng)建對(duì)象的接口。
? ? ? ?先來(lái)看看它的組成:
?
? ? ? ?1) 工廠類角色:這是本模式的核心,含有一定的商業(yè)邏輯和判斷邏輯。在java中它往往由一個(gè)具體類實(shí)現(xiàn)。
?
? ? ? ?2) 抽象產(chǎn)品角色:它一般是具體產(chǎn)品繼承的父類或者實(shí)現(xiàn)的接口。在java中由接口或者抽象類來(lái)實(shí)現(xiàn)。
?
? ? ? ?3) 具體產(chǎn)品角色:工廠類所創(chuàng)建的對(duì)象就是此角色的實(shí)例。在java中由一個(gè)具體類實(shí)現(xiàn)。
?
? ? ? ?來(lái)用類圖來(lái)清晰的表示下的它們之間的關(guān)系(如果對(duì)類圖不太了解,請(qǐng)參考我關(guān)于類圖的文章):?

那么簡(jiǎn)單工廠模式怎么來(lái)使用呢?我們就以簡(jiǎn)單工廠模式來(lái)改造暴發(fā)戶坐車的方式——現(xiàn)在暴發(fā)戶只需要坐在車?yán)飳?duì)司機(jī)說(shuō)句:“開車”就可以了。

//抽象產(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 "); } } 。。。(奧迪我就不寫了:P)
//工廠類角色 public class Driver{ //工廠方法.注意 返回類型為抽象產(chǎn)品角色public static Car driverCar(String s)throws Exception {//判斷邏輯,返回具體的產(chǎn)品角色給Clientif(s.equalsIgnoreCase("Benz")) return new Benz();else if(s.equalsIgnoreCase("Bmw"))return new Bmw();...... else throw new Exception();
//歡迎暴發(fā)戶出場(chǎng)...... public class Magnate{public static void main(String[] args){try{ //告訴司機(jī)我今天坐奔馳 Car car = Driver.driverCar("benz"); //下命令:開車 car.drive(); <pre code_snippet_id="1728121" snippet_file_name="blog_20160624_2_8079212" name="code" class="java"> ...... ?
? ? 將本程序空缺的其他信息填充完整后即可運(yùn)行。如果你將所有的類放在一個(gè)文件中,請(qǐng)不要忘記只能有一個(gè)類被聲明為public。本程序在jdk1.4 下運(yùn)行通過(guò)。
?
? ? ?程序中各個(gè)類的關(guān)系表達(dá)如下:
?
?? ? ? ?這便是簡(jiǎn)單工廠模式了。怎么樣,使用起來(lái)很簡(jiǎn)單吧?那么它帶來(lái)了什么好處呢?
? ? ? ?首先,使用了簡(jiǎn)單工廠模式后,我們的程序不在“有病”,更加符合現(xiàn)實(shí)中的情況;而且客戶端免除了直接創(chuàng)建產(chǎn)品對(duì)象的責(zé)任,而僅僅負(fù)責(zé)“消費(fèi)”產(chǎn)品(正如暴發(fā)戶所為)。
? ? ? ?下面我們從開閉原則(對(duì)擴(kuò)展開放;對(duì)修改封閉)上來(lái)分析下簡(jiǎn)單工廠模式。當(dāng)暴發(fā)戶增加了一輛車的時(shí)候,只要符合抽象產(chǎn)品制定的合同,那么只要通知工廠類知道就可以被客戶使用了。所以對(duì)產(chǎn)品部分來(lái)說(shuō),它是符合開閉原則的;但是工廠部分好像不太理想,因?yàn)槊吭黾右惠v車,都要在工廠類中增加相應(yīng)的業(yè)務(wù)邏輯或者判斷邏輯,這顯然是違背開閉原則的。可想而知對(duì)于新產(chǎn)品的加入,工廠類是很被動(dòng)的。對(duì)于這樣的工廠類(在我們的例子中是為司機(jī)師傅),我們稱它為全能類或者上帝類。
? ? ? ?我們舉的例子是最簡(jiǎn)單的情況,而在實(shí)際應(yīng)用中,很可能產(chǎn)品是一個(gè)多層次的樹狀結(jié)構(gòu)。由于簡(jiǎn)單工廠模式中只有一個(gè)工廠類來(lái)對(duì)應(yīng)這些產(chǎn)品,所以這可能會(huì)把我們的上帝累壞了,也累壞了我們這些程序員:(
? ? ? ?于是工廠方法模式作為救世主出現(xiàn)了。?

四、工廠方法模式
?? ? ? ?工廠方法模式去掉了簡(jiǎn)單工廠模式中工廠方法的靜態(tài)屬性,使得它可以被子類繼承。這樣在簡(jiǎn)單工廠模式里集中在工廠方法上的壓力可以由工廠方法模式里不同的工廠子類來(lái)分擔(dān)。
?
? ? ? ?你應(yīng)該大致猜出了工廠方法模式的結(jié)構(gòu),來(lái)看下它的組成:
?
? ? ? ?1) ?抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無(wú)關(guān)。是具體工廠角色必須實(shí)現(xiàn)的接口或者必須繼承的父類。在java中它由抽象類或者接口來(lái)實(shí)現(xiàn)。
?
? ? ? ?2) ?具體工廠角色:它含有和具體業(yè)務(wù)邏輯有關(guān)的代碼。由應(yīng)用程序調(diào)用以創(chuàng)建對(duì)應(yīng)的具體產(chǎn)品的對(duì)象。
?
? ? ? ?3) ?抽象產(chǎn)品角色:它是具體產(chǎn)品繼承的父類或者是實(shí)現(xiàn)的接口。在java中一般有抽象類或者接口來(lái)實(shí)現(xiàn)。
?
? ? ? ?4) ?具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對(duì)象就是此角色的實(shí)例。在java中由具體的類來(lái)實(shí)現(xiàn)。
?
? ? ? ?用類圖來(lái)清晰的表示下的它們之間的關(guān)系:
?
??
? ? ? ?工廠方法模式使用繼承自抽象工廠角色的多個(gè)子類來(lái)代替簡(jiǎn)單工廠模式中的“上帝類”。正如上面所說(shuō),這樣便分擔(dān)了對(duì)象承受的壓力;而且這樣使得結(jié)構(gòu)變得靈活起來(lái)——當(dāng)有新的產(chǎn)品(即暴發(fā)戶的汽車)產(chǎn)生時(shí),只要按照抽象產(chǎn)品角色、抽象工廠角色提供的合同來(lái)生成,那么就可以被客戶使用,而不必去修改任何已有的代碼。可以看出工廠角色的結(jié)構(gòu)也是符合開閉原則的!
? ? ? ?我們還是老規(guī)矩,使用一個(gè)完整的例子來(lái)看看工廠模式各個(gè)角色之間是如何來(lái)協(xié)調(diào)的。話說(shuō)暴發(fā)戶生意越做越大,自己的愛車也越來(lái)越多。這可苦了那位司機(jī)師傅了,什么車它都要記得,維護(hù),都要經(jīng)過(guò)他來(lái)使用!于是暴發(fā)戶同情他說(shuō):看你跟我這么多年的份上,以后你不用這么辛苦了,我給你分配幾個(gè)人手,你只管管好他們就行了!于是,工廠方法模式的管理出現(xiàn)了。代碼如下:
//抽象產(chǎn)品角色,具體產(chǎn)品角色與簡(jiǎn)單工廠模式類似,只是變得復(fù)雜了些,這里略。 //抽象工廠角色 public interface Driver{public Car driverCar(); } public class BenzDriver implements Driver{public Car driverCar(){return new Benz();} } public class BmwDriver implements Driver{public Car driverCar() {return new Bmw(); } }//應(yīng)該和具體產(chǎn)品形成對(duì)應(yīng)關(guān)系... //有請(qǐng)暴發(fā)戶先生public class Magnate{public static void main(String[] args){try{ Driver driver = new BenzDriver();Car car = driver.driverCar();car.drive();}……}? ? ? 可以看出工廠方法的加入,使得對(duì)象的數(shù)量成倍增長(zhǎng)。當(dāng)產(chǎn)品種類非常多時(shí),會(huì)出現(xiàn)大量的與之對(duì)應(yīng)的工廠對(duì)象,這不是我們所希望的。因?yàn)槿绻荒鼙苊膺@種情況,可以考慮使用簡(jiǎn)單工廠模式與工廠方法模式相結(jié)合的方式來(lái)減少工廠類:即對(duì)于產(chǎn)品樹上類似的種類(一般是樹的葉子中互為兄弟的)使用簡(jiǎn)單工廠模式來(lái)實(shí)現(xiàn)。
??
五、小結(jié)
?? ? ??工廠方法模式仿佛已經(jīng)很完美的對(duì)對(duì)象的創(chuàng)建進(jìn)行了包裝,使得客戶程序中僅僅處理抽象產(chǎn)品角色提供的接口。那我們是否一定要在代碼中遍布工廠呢?大可不必。也許在下面情況下你可以考慮使用工廠方法模式:
?
? ? ??1) 當(dāng)客戶程序不需要知道要使用對(duì)象的創(chuàng)建過(guò)程。
? ? ??2)客戶程序使用的對(duì)象存在變動(dòng)的可能,或者根本就不知道使用哪一個(gè)具體的對(duì)象。?
? ? ??簡(jiǎn)單工廠模式與工廠方法模式真正的避免了代碼的改動(dòng)了?沒有。在簡(jiǎn)單工廠模式中,新產(chǎn)品的加入要修改工廠角色中的判斷語(yǔ)句;而在工廠方法模式中,要么將判斷邏輯留在抽象工廠角色中,要么在客戶程序中將具體工廠角色寫死(就象上面的例子一樣)。而且產(chǎn)品對(duì)象創(chuàng)建條件的改變必然會(huì)引起工廠角色的修改。
? ? ? ?面對(duì)這種情況,Java的反射機(jī)制與配置文件的巧妙結(jié)合突破了限制——這在Spring中完美的體現(xiàn)了出來(lái)。
?
六、抽象工廠模式
? ? ? 先來(lái)認(rèn)識(shí)下什么是產(chǎn)品族: 位于不同產(chǎn)品等級(jí)結(jié)構(gòu)中,功能相關(guān)聯(lián)的產(chǎn)品組成的家族。還是讓我們用一個(gè)例子來(lái)形象地說(shuō)明一下吧。
?
? ? ??圖中的BmwCar和BenzCar就是兩個(gè)產(chǎn)品樹(產(chǎn)品層次結(jié)構(gòu));而如圖所示的BenzSportsCar和BmwSportsCar就是一個(gè)產(chǎn)品族。他們都可以放到跑車家族中,因此功能有所關(guān)聯(lián)。同理BmwBussinessCar和BenzSportsCar也是一個(gè)產(chǎn)品族。
? ? ? 回到抽象工廠模式的話題上。可以說(shuō),抽象工廠模式和工廠方法模式的區(qū)別就在于需要?jiǎng)?chuàng)建對(duì)象的復(fù)雜程度上。而且抽象工廠模式是三個(gè)里面最為抽象、最具一般性的。抽象工廠模式的用意為:給客戶端提供一個(gè)接口,可以創(chuàng)建多個(gè)產(chǎn)品族中的產(chǎn)品對(duì)象,而且使用抽象工廠模式還要滿足一下條件:
?
? ? ??1) 系統(tǒng)中有多個(gè)產(chǎn)品族,而系統(tǒng)一次只可能消費(fèi)其中一族產(chǎn)品。
? ? ??2) 同屬于同一個(gè)產(chǎn)品族的產(chǎn)品以其使用。
? ? ??來(lái)看看抽象工廠模式的各個(gè)角色(和工廠方法的如出一轍):
? ? ??1) 抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無(wú)關(guān)。是具體工廠角色必須實(shí)現(xiàn)的接口或者必須繼承的父類。在java中它由抽象類或者接口來(lái)實(shí)現(xiàn)。
? ? ??2) 具體工廠角色:它含有和具體業(yè)務(wù)邏輯有關(guān)的代碼。由應(yīng)用程序調(diào)用以創(chuàng)建對(duì)應(yīng)的具體產(chǎn)品的對(duì)象。在java中它由具體的類來(lái)實(shí)現(xiàn)。
? ? ??3)抽象產(chǎn)品角色:它是具體產(chǎn)品繼承的父類或者是實(shí)現(xiàn)的接口。在java中一般有抽象類或者接口來(lái)實(shí)現(xiàn)。
? ? ??4) 具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對(duì)象就是此角色的實(shí)例。在java中由具體的類來(lái)實(shí)現(xiàn)。
?
類圖如下:
?
?看過(guò)了前兩個(gè)模式,對(duì)這個(gè)模式各個(gè)角色之間的協(xié)調(diào)情況應(yīng)該心里有個(gè)數(shù)了,我就不舉具體的例子了。只是一定要注意滿足使用抽象工廠模式的條件哦。
抽象工廠示例:
Java代碼 收藏代碼 // 產(chǎn)品 Plant接口 public interface Plant { }//標(biāo)志接口 //具體產(chǎn)品PlantA,PlantB public class PlantA implements Plant { public PlantA () { System.out.println("create PlantA !"); } public void doSomething() { System.out.println(" PlantA do something ..."); } } public class PlantB implements Plant { public PlantB () { System.out.println("create PlantB !"); } public void doSomething() { System.out.println(" PlantB do something ..."); } } // 產(chǎn)品 Fruit接口 public interface Fruit { } //具體產(chǎn)品FruitA,FruitB public class FruitA implements Fruit { public FruitA() { System.out.println("create FruitA !"); } public void doSomething() { System.out.println(" FruitA do something ..."); } } public class FruitB implements Fruit { public FruitB() { System.out.println("create FruitB !"); } public void doSomething() { System.out.println(" FruitB do something ..."); } } // 抽象工廠方法 public interface AbstractFactory { public Plant createPlant(); public Fruit createFruit(); } //具體工廠方法 public class FactoryA implements AbstractFactory { public Plant createPlant() { return new PlantA(); } public Fruit createFruit() { return new FruitA(); } } public class FactoryB implements AbstractFactory { public Plant createPlant() { return new PlantB(); } public Fruit createFruit() { return new FruitB(); } } Java代碼 收藏代碼 //調(diào)用工廠方法 public Client { public method1() { AbstractFactory instance = new FactoryA(); instance.createPlant(); } }
? ? ? 抽象工廠模式與工廠方法模式的區(qū)別 :可以這么說(shuō),工廠方法模式是一種極端情況的抽象工廠模式,而抽象工廠模式可以看成是工廠方法模式的一種推廣。
? ? ??(1)、其實(shí)工廠方法模式是用來(lái)創(chuàng)建一個(gè)產(chǎn)品的等級(jí)結(jié)構(gòu)的,而抽象工廠模式是用來(lái)創(chuàng)建多個(gè)產(chǎn)品的等級(jí)結(jié)構(gòu)的。工廠方法創(chuàng)建一般只有一個(gè)方法,創(chuàng)建一種產(chǎn)品。抽象工廠一般有多個(gè)方法,創(chuàng)建一系列產(chǎn)品。?
? ? ??(2)、工廠方法模式只有一個(gè)抽象產(chǎn)品類,而抽象工廠模式有多個(gè)。工廠方法模式的具體工廠類只能創(chuàng)建一個(gè)具體產(chǎn)品類的實(shí)例,而抽象工廠模式可以創(chuàng)建多個(gè)。?
簡(jiǎn)而言之->?

(1)工廠方法模式:

? ? ??一個(gè)抽象產(chǎn)品類,可以派生出多個(gè)具體產(chǎn)品類。 ??

? ? ??一個(gè)抽象工廠類,可以派生出多個(gè)具體工廠類。 ??
? ? ??每個(gè)具體工廠類只能創(chuàng)建一個(gè)具體產(chǎn)品類的實(shí)例。 ??

(2)抽象工廠模式:

? ? ??多個(gè)抽象產(chǎn)品類,每個(gè)抽象產(chǎn)品類可以派生出多個(gè)具體產(chǎn)品類。 ??

? ? ??一個(gè)抽象工廠類,可以派生出多個(gè)具體工廠類。 ??
? ? ??每個(gè)具體工廠類可以創(chuàng)建多個(gè)具體產(chǎn)品類的實(shí)例。

總結(jié)

以上是生活随笔為你收集整理的工厂方法模式与抽象工厂模式的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。