PHP設(shè)計(jì)模式(一)工廠模式Factory實(shí)例詳解【創(chuàng)建型】
本文實(shí)例講述了PHP設(shè)計(jì)模式(一)工廠模式Factory。分享給大家供大家參考,具體如下:
在面向?qū)ο缶幊讨? 最通常的方法是一個(gè)new操作符產(chǎn)生一個(gè)對(duì)象實(shí)例,new操作符就是用來(lái)構(gòu)造對(duì)象實(shí)例的。但是在一些情況下, new操作符直接生成對(duì)象會(huì)帶來(lái)一些問(wèn)題。舉例來(lái)說(shuō), 許多類(lèi)型對(duì)象的創(chuàng)造需要一系列的步驟: 你可能需要計(jì)算或取得對(duì)象的初始設(shè)置; 選擇生成哪個(gè)子對(duì)象實(shí)例; 或在生成你需要的對(duì)象之前必須先生成一些輔助功能的對(duì)象。 在這些情況,新對(duì)象的建立就是一個(gè) “過(guò)程”,不僅是一個(gè)操作,像一部大機(jī)器中的一個(gè)齒輪傳動(dòng)。
模式的問(wèn)題:你如何能輕松方便地構(gòu)造對(duì)象實(shí)例,而不必關(guān)心構(gòu)造對(duì)象實(shí)例的細(xì)節(jié)和復(fù)雜過(guò)程呢?
解決方案:建立一個(gè)工廠來(lái)創(chuàng)建對(duì)象。
實(shí)現(xiàn):
一、引言1)還沒(méi)有工廠時(shí)代:假如還沒(méi)有工業(yè)革命,如果一個(gè)客戶要一款寶馬車(chē),一般的做法是客戶去創(chuàng)建一款寶馬車(chē),然后拿來(lái)用。 2)簡(jiǎn)單工廠模式:后來(lái)出現(xiàn)工業(yè)革命。用戶不用去創(chuàng)建寶馬車(chē)。因?yàn)榭蛻粲幸粋€(gè)工廠來(lái)幫他創(chuàng)建寶馬.想要什么車(chē),這個(gè)工廠就可以建。比如想要320i系列車(chē)。工廠就創(chuàng)建這個(gè)系列的車(chē)。即工廠可以創(chuàng)建產(chǎn)品。 3)工廠方法模式時(shí)代:為了滿足客戶,寶馬車(chē)系列越來(lái)越多,如320i,523i,30li等系列一個(gè)工廠無(wú)法創(chuàng)建所有的寶馬系列。于是由單獨(dú)分出來(lái)多個(gè)具體的工廠。每個(gè)具體工廠創(chuàng)建一種系列。即具體工廠類(lèi)只能創(chuàng)建一個(gè)具體產(chǎn)品。但是寶馬工廠還是個(gè)抽象。你需要指定某個(gè)具體的工廠才能生產(chǎn)車(chē)出來(lái)。 4)抽象工廠模式時(shí)代:隨著客戶的要求越來(lái)越高,寶馬車(chē)必須配置空調(diào)。而且這空調(diào)必須對(duì)應(yīng)給系列車(chē)才能使用。于是這個(gè)工廠開(kāi)始生產(chǎn)寶馬車(chē)和需要的空調(diào)。 最終是客戶只要對(duì)寶馬的銷(xiāo)售員說(shuō):我要523i空調(diào)車(chē),銷(xiāo)售員就直接給他523i空調(diào)車(chē)了。而不用自己去創(chuàng)建523i空調(diào)車(chē)寶馬車(chē). (我只是舉個(gè)例子,說(shuō)到寶馬配置空調(diào)完全是為了舉例,甚至有點(diǎn)扯,哪有車(chē)和空調(diào)必須對(duì)應(yīng)才能使用?。? 這就是工廠模式。
二、分類(lèi)工廠模式主要是為創(chuàng)建對(duì)象提供過(guò)渡接口,以便將創(chuàng)建對(duì)象的具體過(guò)程屏蔽隔離起來(lái),達(dá)到提高靈活性的目的。 工廠模式可以分為三類(lèi): 1)簡(jiǎn)單工廠模式(Simple Factory) 2)工廠方法模式(Factory Method) 3)抽象工廠模式(Abstract Factory) 這三種模式從上到下逐步抽象,并且更具一般性。 GOF在《設(shè)計(jì)模式》一書(shū)中將工廠模式分為兩類(lèi):工廠方法模式(Factory Method)與抽象工廠模式(Abstract Factory)。將簡(jiǎn)單工廠模式(Simple Factory)看為工廠方法模式的一種特例,兩者歸為一類(lèi)。
三、區(qū)別工廠方法模式:一個(gè)抽象產(chǎn)品類(lèi),可以派生出多個(gè)具體產(chǎn)品類(lèi)。 一個(gè)抽象工廠類(lèi),可以派生出多個(gè)具體工廠類(lèi)。 每個(gè)具體工廠類(lèi)只能創(chuàng)建一個(gè)具體產(chǎn)品類(lèi)的實(shí)例。
抽象工廠模式:多個(gè)抽象產(chǎn)品類(lèi),每個(gè)抽象產(chǎn)品類(lèi)可以派生出多個(gè)具體產(chǎn)品類(lèi)。 一個(gè)抽象工廠類(lèi),可以派生出多個(gè)具體工廠類(lèi)。每個(gè)具體工廠類(lèi)可以創(chuàng)建多個(gè)具體產(chǎn)品類(lèi)的實(shí)例。
區(qū)別:工廠方法模式只有一個(gè)抽象產(chǎn)品類(lèi),而抽象工廠模式有多個(gè)。 工廠方法模式的具體工廠類(lèi)只能創(chuàng)建一個(gè)具體產(chǎn)品類(lèi)的實(shí)例,而抽象工廠模式可以創(chuàng)建多個(gè)。兩者皆可。
四、簡(jiǎn)單工廠模式建立一個(gè)工廠(一個(gè)函數(shù)或一個(gè)類(lèi)方法)來(lái)制造新的對(duì)象。
分布說(shuō)明引子:從無(wú)到有。客戶自己創(chuàng)建寶馬車(chē),然后拿來(lái)用。
<?php/** * 車(chē)子系列 * */Class BWM320{function __construct($pa) { }}Class BMW523{ function __construc($pb){}}/** * * 客戶自己創(chuàng)建寶馬車(chē) */class Customer { function createBMW320(){ return new BWM320(); } function createBMW523(){ return new BMW523(); }}
客戶需要知道怎么去創(chuàng)建一款車(chē),客戶和車(chē)就緊密耦合在一起了.為了降低耦合,就出現(xiàn)了工廠類(lèi),把創(chuàng)建寶馬的操作細(xì)節(jié)都放到了工廠里面去,客戶直接使用工廠的創(chuàng)建工廠方法,傳入想要的寶馬車(chē)型號(hào)就行了,而不必去知道創(chuàng)建的細(xì)節(jié).這就是工業(yè)革命了:簡(jiǎn)單工廠模式
即我們建立一個(gè)工廠類(lèi)方法來(lái)制造新的對(duì)象。如圖:
產(chǎn)品類(lèi):
<?php/** * 車(chē)子系列 * */abstract Class BWM{ function __construct($pa) { }}Class BWM320 extends BWM{ function __construct($pa) { }}Class BMW523 extends BWM{ function __construc($pb){ }}
工廠類(lèi):
/** * * 工廠創(chuàng)建車(chē) */class Factory { static function createBMW($type){ switch ($type) { case 320: return new BWM320(); case 523: return new BMW523(); //.... }}
客戶類(lèi):
/** * * 客戶通過(guò)工廠獲取車(chē) */class Customer { private $BMW; function getBMW($type){ $this¬-> BMW = Factory::createBMW($type); }}
簡(jiǎn)單工廠模式又稱(chēng)靜態(tài)工廠方法模式。重命名上就可以看出這個(gè)模式一定很簡(jiǎn)單。它存在的目的很簡(jiǎn)單:定義一個(gè)用于創(chuàng)建對(duì)象的接口。
先來(lái)看看它的組成: 1) 工廠類(lèi)角色:這是本模式的核心,含有一定的商業(yè)邏輯和判斷邏輯。 2) 抽象產(chǎn)品角色:它一般是具體產(chǎn)品繼承的父類(lèi)或者實(shí)現(xiàn)的接口。 3) 具體產(chǎn)品角色:工廠類(lèi)所創(chuàng)建的對(duì)象就是此角色的實(shí)例。在java中由一個(gè)具體類(lèi)實(shí)現(xiàn)。
下面我們從開(kāi)閉原則(對(duì)擴(kuò)展開(kāi)放;對(duì)修改封閉)上來(lái)分析下簡(jiǎn)單工廠模式。當(dāng)客戶不再滿足現(xiàn)有的車(chē)型號(hào)的時(shí)候,想要一種速度快的新型車(chē),只要這種車(chē)符合抽象產(chǎn)品制定的合同,那么只要通知工廠類(lèi)知道就可以被客戶使用了。所以對(duì)產(chǎn)品部分來(lái)說(shuō),它是符合開(kāi)閉原則的;但是工廠部分好像不太理想,因?yàn)槊吭黾右环N新型車(chē),都要在工廠類(lèi)中增加相應(yīng)的創(chuàng)建業(yè)務(wù)邏輯(createBMW($type)方法需要新增case),這顯然是違背開(kāi)閉原則的。可想而知對(duì)于新產(chǎn)品的加入,工廠類(lèi)是很被動(dòng)的。對(duì)于這樣的工廠類(lèi),我們稱(chēng)它為全能類(lèi) 或者上帝類(lèi)。 我們舉的例子是最簡(jiǎn)單的情況,而在實(shí)際應(yīng)用中,很可能產(chǎn)品是一個(gè)多層次的樹(shù)狀結(jié)構(gòu)。由于簡(jiǎn)單工廠模式中只有一個(gè)工廠類(lèi)來(lái)對(duì)應(yīng)這些產(chǎn)品,所以這可能會(huì)把我們的上帝累壞了,也累壞了我們這些程序員:( 于是工廠方法模式作為救世主出現(xiàn)了。 工廠類(lèi)定義成了接口,而每新增的車(chē)種類(lèi)型,就增加該車(chē)種類(lèi)型對(duì)應(yīng)工廠類(lèi)的實(shí)現(xiàn),這樣工廠的設(shè)計(jì)就可以擴(kuò)展了,而不必去修改原來(lái)的代碼。
五、工廠方法模式工廠方法模式去掉了簡(jiǎn)單工廠模式中工廠方法的靜態(tài)屬性,使得它可以被子類(lèi)繼承。這樣在簡(jiǎn)單工廠模式里集中在工廠方法上的壓力可以由工廠方法模式里不同的工廠子類(lèi)來(lái)分擔(dān)。 工廠方法模式組成:
1)抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無(wú)關(guān)。是具體工廠角色必須實(shí)現(xiàn)的接口或者必須繼承的父類(lèi)。在java中它由抽象類(lèi)或者接口來(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)品繼承的父類(lèi)或者是實(shí)現(xiàn)的接口。在java中一般有抽象類(lèi)或者接口來(lái)實(shí)現(xiàn)。4)具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對(duì)象就是此角色的實(shí)例。在java中由具體的類(lèi)來(lái)實(shí)現(xiàn)。工廠方法模式使用繼承自抽象工廠角色的多個(gè)子類(lèi)來(lái)代替簡(jiǎn)單工廠模式中的“上帝類(lèi)”。正如上面所說(shuō),這樣便分擔(dān)了對(duì)象承受的壓力;而且這樣使得結(jié)構(gòu)變得靈活 起來(lái)——當(dāng)有新的產(chǎn)品產(chǎn)生時(shí),只要按照抽象產(chǎn)品角色、抽象工廠角色提供的合同來(lái)生成,那么就可以被客戶使用,而不必去修改任何已有 的代碼。可以看出工廠角色的結(jié)構(gòu)也是符合開(kāi)閉原則的!
代碼如下:
產(chǎn)品類(lèi):
<?php/** * 車(chē)子系列 * */abstract Class BWM{function __construct($pa) {}}Class BWM320 extends BWM{function __construct($pa) {}}Class BMW523 extends BWM{ function __construc($pb){}}
創(chuàng)建工廠類(lèi):
/** * 創(chuàng)建工廠的接口 * */interface FactoryBMW { function createBMW(); } /** * * 創(chuàng)建BWM320車(chē) */class FactoryBWM320 implements FactoryBMW { function createBMW($type){ return new BWM320(); }}/** * * 創(chuàng)建BWM523車(chē) */class FactoryBWM523 implements FactoryBMW { function createBMW($type){ return new BMW523(); }}
客戶類(lèi):
/** * * 客戶得到車(chē) */class Customer { private $BMW; function getBMW($type){ switch ($type) { case 320: $BWM320 = new FactoryBWM320(); return $BWM320->createBMW(); case 523: $BWM523 = new FactoryBWM523(); return $BWM320->createBMW(); //.... } }}
可以看出工廠方法的加入,使得對(duì)象的數(shù)量成倍增長(zhǎng)。當(dāng)產(chǎn)品種類(lèi)非常多時(shí),會(huì)出現(xiàn)大量的與之對(duì)應(yīng)的工廠對(duì)象,這不是我們所希望的。因?yàn)槿绻荒鼙苊膺@種情 況,可以考慮使用簡(jiǎn)單工廠模式與工廠方法模式相結(jié)合的方式來(lái)減少工廠類(lèi):即對(duì)于產(chǎn)品樹(shù)上類(lèi)似的種類(lèi)(一般是樹(shù)的葉子中互為兄弟的)使用簡(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)了?沒(méi)有。在簡(jiǎn)單工廠模式中,新產(chǎn)品的加入要修改工廠角色中的判斷語(yǔ)句;而在工廠方法模式中,要么將判 斷邏輯留在抽象工廠角色中,要么在客戶程序中將具體工廠角色寫(xiě)死(就象上面的例子一樣)。而且產(chǎn)品對(duì)象創(chuàng)建條件的改變必然會(huì)引起工廠角色的修改。 面對(duì)這種情況,我們可以使用反射機(jī)制:
class Customer { private $BMW; function getBMW($type){ $class = new ReflectionClass(’FactoryBWM’ .$type );//建立 ’FactoryBWM’這個(gè)類(lèi)的反射類(lèi) $instance = $class->newInstanceArgs();//相當(dāng)于實(shí)例化’FactoryBWM’ .$type類(lèi) return $instance->createBMW(); //或者直接 /** * $instance = new ’FactoryBWM’ .$type(); * return $instance->createBMW(); */ }}六、抽象工廠模式
隨著客戶的要求越來(lái)越高,寶馬車(chē)需要配置空調(diào)。于是這個(gè)工廠開(kāi)始生產(chǎn)寶馬車(chē)和配置需要的空調(diào)。這時(shí)候工廠有二個(gè)系列的產(chǎn)品:寶馬車(chē)和空調(diào).寶馬車(chē)必須使用對(duì)應(yīng)的空調(diào)才能使用.這時(shí)候分別使用一個(gè)車(chē)工廠和一個(gè)空調(diào)工廠都不能滿足我們的需求,我們必須確認(rèn)車(chē)跟空調(diào)的對(duì)應(yīng)關(guān)系。因此把車(chē)工廠跟空調(diào)工廠聯(lián)系在一起。因此出現(xià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)品以其使用。
抽象工廠模式的各個(gè)角色(和工廠方法一樣):
1)抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無(wú)關(guān)。是具體工廠角色必須實(shí)現(xiàn)的接口或者必須繼承的父類(lèi)。在java中它由抽象類(lèi)或者接口來(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)品繼承的父類(lèi)或者是實(shí)現(xiàn)的接口。 4)具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對(duì)象就是此角色的實(shí)例。
其結(jié)構(gòu):
我們的例子:
代碼:
產(chǎn)品類(lèi):
<?php/** * 車(chē)子系列以及型號(hào) * */abstract class BWM{}class BWM523 extends BWM {}class BWM320 extends BWM {}/** * 空調(diào) * */abstract class aircondition{}class airconditionBWM320 extends aircondition {}class airconditionBWM52 extends aircondition {}
創(chuàng)建工廠類(lèi):
/** * 創(chuàng)建工廠的接口 * */interface FactoryBMW { function createBMW(); function createAirC(); } /** * * 創(chuàng)建BWM320車(chē) */class FactoryBWM320 implements FactoryBMW { function createBMW(){ return new BWM320();}function createAirC(){ //空調(diào) return new airconditionBWM320();}}/** * * 創(chuàng)建BWM523車(chē) */class FactoryBWM523 implements FactoryBMW { function createBMW(){ return new BWM523();}function createAirC(){ return new airconditionBWM523();}}
客戶:
/** * * 客戶得到車(chē) */class Customer { private $BMW; private $airC; function getBMW($type){ $class = new ReflectionClass(’FactoryBWM’ .$type );//建立 Person這個(gè)類(lèi)的反射類(lèi) $instance = $class->newInstanceArgs();//相當(dāng)于實(shí)例化Person 類(lèi) $this->BMW = $instance->createBMW(); $this->airC = $instance->createAirC(); }}
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門(mén)教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP基本語(yǔ)法入門(mén)教程》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫(kù)操作入門(mén)教程》及《php常見(jiàn)數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。
相關(guān)文章:
1. ASP.NET Core實(shí)現(xiàn)中間件的幾種方式2. jsp+servlet簡(jiǎn)單實(shí)現(xiàn)上傳文件功能(保存目錄改進(jìn))3. html中的form不提交(排除)某些input 原創(chuàng)4. 利用CSS制作3D動(dòng)畫(huà)5. XML解析錯(cuò)誤:未組織好 的解決辦法6. 告別AJAX實(shí)現(xiàn)無(wú)刷新提交表單7. jsp實(shí)現(xiàn)登錄驗(yàn)證的過(guò)濾器8. jsp實(shí)現(xiàn)局部刷新頁(yè)面、異步加載頁(yè)面的方法9. 使用css實(shí)現(xiàn)全兼容tooltip提示框10. .Net Core和RabbitMQ限制循環(huán)消費(fèi)的方法
