PHP設(shè)計(jì)模式之抽象工廠模式工廠模式系列中的重頭戲來(lái)了,沒(méi)錯(cuò),那正是傳聞中的抽象工廠模式。初次聽(tīng)到這個(gè)名字的時(shí)候你有什么感覺(jué)?反正我是感覺(jué)這貨應(yīng)該是非常高大上的,畢竟包含著“抽象”兩個(gè)字。話說(shuō)這兩個(gè)字在開(kāi)發(fā)中真的是有點(diǎn)高大上的感覺(jué),一帶上抽象兩字就好像哪哪都很厲害了呢。不過(guò),抽象工廠也確實(shí)可以說(shuō)是工廠模式的大哥大。 Gof類(lèi)圖及解釋其實(shí)只要理解了工廠方法模式,就很容易明白抽象工廠模式。怎么說(shuō)呢?還是一樣的延遲到子類(lèi),還是一樣的返回指定的對(duì)象。只是抽象工廠里面不僅僅只返回一個(gè)對(duì)象,而是返回一堆。 GoF定義:提供一個(gè)創(chuàng)建一系列相關(guān)或相互依賴對(duì)象的接口,而無(wú)需指定它們具體的類(lèi)。 GoF類(lèi)圖
 左邊是兩個(gè)工廠1和2,都繼承一個(gè)抽象工廠,都實(shí)現(xiàn)了CreateProductA和CreateProductB方法 工廠1生產(chǎn)的是ProductA1和ProductB1 同樣的,工廠2生產(chǎn)的是ProductA2和ProductB2
代碼實(shí)現(xiàn)
// 商品A抽象接口 interface AbstractProductA { public function show(): void; }
// 商品A1實(shí)現(xiàn) class ProductA1 implements AbstractProductA { public function show(): void { echo 'ProductA1 is Show!' . PHP_EOL; } } // 商品A2實(shí)現(xiàn) class ProductA2 implements AbstractProductA { public function show(): void { echo 'ProductA2 is Show!' . PHP_EOL; } }
// 商品B抽象接口 interface AbstractProductB { public function show(): void; } // 商品B1實(shí)現(xiàn) class ProductB1 implements AbstractProductB { public function show(): void { echo 'ProductB1 is Show!' . PHP_EOL; } } // 商品B2實(shí)現(xiàn) class ProductB2 implements AbstractProductB { public function show(): void { echo 'ProductB2 is Show!' . PHP_EOL; } }
商品的實(shí)現(xiàn),東西很多吧,這回其實(shí)是有四件商品了分別是A1、A2、B1和B2,他們之間假設(shè)有這樣的關(guān)系,A1和B1是同類(lèi)相關(guān)的商品,B1和B2是同類(lèi)相關(guān)的商品 // 抽象工廠接口 interface AbstractFactory { // 創(chuàng)建商品A public function CreateProductA(): AbstractProductA; // 創(chuàng)建商品B public function CreateProductB(): AbstractProductB; }
// 工廠1,實(shí)現(xiàn)商品A1和商品B1 class ConcreteFactory1 implements AbstractFactory { public function CreateProductA(): AbstractProductA { return new ProductA1(); } public function CreateProductB(): AbstractProductB { return new ProductB1(); } }
// 工廠2,實(shí)現(xiàn)商品A2和商品B2 class ConcreteFactory2 implements AbstractFactory { public function CreateProductA(): AbstractProductA { return new ProductA2(); } public function CreateProductB(): AbstractProductB { return new ProductB2(); } }
而我們的工廠也是工廠1和工廠2,工廠1生產(chǎn)的是A1和B1這兩種相關(guān)聯(lián)的產(chǎn)品,工廠2生產(chǎn)的是A2和B2這兩種商品。好吧,我知道這里還是有點(diǎn)抽象,可能還是搞不懂為什么要這樣,我們繼續(xù)以手機(jī)生產(chǎn)來(lái)舉例。 我們的手機(jī)品牌起來(lái)了,所以周邊如手機(jī)膜、手機(jī)殼也交給了富X康(AbstractFactory)來(lái)幫我搞定。上回說(shuō)到,我已經(jīng)有幾款不同類(lèi)型的手機(jī)了,于是還是按原來(lái)那樣,衡陽(yáng)工廠(Factory1)生產(chǎn)型號(hào)1001的手機(jī)(ProductA1),同時(shí)型號(hào)1001手機(jī)的手機(jī)膜(ProductB1)和手機(jī)殼(ProductC1)也是衡陽(yáng)工廠生產(chǎn)出來(lái)。而型號(hào)1002的手機(jī)(ProductA2)還是在鄭州工廠(Factory2),這個(gè)型號(hào)的手機(jī)膜(ProductB2)和手機(jī)膜(ProductC2)也就交給他們來(lái)搞定吧。于是,我還是只去跟總廠下單,他們讓不同的工廠給我生產(chǎn)了一整套的手機(jī)產(chǎn)品,可以直接賣(mài)套裝咯??! 完整代碼:抽象工廠模式 實(shí)例是不是看得還是有點(diǎn)暈。其實(shí)說(shuō)簡(jiǎn)單點(diǎn),真的就是在一個(gè)工廠類(lèi)中通過(guò)不同的方法返回不同的對(duì)象而已。讓我們?cè)俅斡冒l(fā)短信的實(shí)例來(lái)講解吧! 場(chǎng)景:這次我們有個(gè)業(yè)務(wù)需求是,不僅要發(fā)短信,還要同時(shí)發(fā)一條推送。短信的目的是通知用戶有新的活動(dòng)參加,而推送不僅通知有新的活動(dòng),直接點(diǎn)擊就可以進(jìn)去領(lǐng)紅包了,是不是很興奮。還好之前我們的選擇的云服務(wù)供應(yīng)商都是即有短信也有推送接口的,所以我們就直接用抽象工廠來(lái)實(shí)現(xiàn)吧! 短信發(fā)送類(lèi)圖
 <?php
interface Message { public function send(string $msg); }
class AliYunMessage implements Message{ public function send(string $msg){ // 調(diào)用接口,發(fā)送短信 // xxxxx return '阿里云短信(原阿里大魚(yú))發(fā)送成功!短信內(nèi)容:' . $msg; } }
class BaiduYunMessage implements Message{ public function send(string $msg){ // 調(diào)用接口,發(fā)送短信 // xxxxx return '百度SMS短信發(fā)送成功!短信內(nèi)容:' . $msg; } }
class JiguangMessage implements Message{ public function send(string $msg){ // 調(diào)用接口,發(fā)送短信 // xxxxx return '極光短信發(fā)送成功!短信內(nèi)容:' . $msg; } }
interface Push { public function send(string $msg); }
class AliYunPush implements Push{ public function send(string $msg){ // 調(diào)用接口,發(fā)送客戶端推送 // xxxxx return '阿里云Android&iOS推送發(fā)送成功!推送內(nèi)容:' . $msg; } }
class BaiduYunPush implements Push{ public function send(string $msg){ // 調(diào)用接口,發(fā)送客戶端推送 // xxxxx return '百度Android&iOS云推送發(fā)送成功!推送內(nèi)容:' . $msg; } }
class JiguangPush implements Push{ public function send(string $msg){ // 調(diào)用接口,發(fā)送客戶端推送 // xxxxx return '極光推送發(fā)送成功!推送內(nèi)容:' . $msg; } }
interface MessageFactory{ public function createMessage(); public function createPush(); }
class AliYunFactory implements MessageFactory{ public function createMessage(){ return new AliYunMessage(); } public function createPush(){ return new AliYunPush(); } }
class BaiduYunFactory implements MessageFactory{ public function createMessage(){ return new BaiduYunMessage(); } public function createPush(){ return new BaiduYunPush(); } }
class JiguangFactory implements MessageFactory{ public function createMessage(){ return new JiguangMessage(); } public function createPush(){ return new JiguangPush(); } }
// 當(dāng)前業(yè)務(wù)需要使用阿里云 $factory = new AliYunFactory(); // $factory = new BaiduYunFactory(); // $factory = new JiguangFactory(); $message = $factory->createMessage(); $push = $factory->createPush(); echo $message->send('您已經(jīng)很久沒(méi)有登錄過(guò)系統(tǒng)了,記得回來(lái)哦!'); echo $push->send('您有新的紅包已到帳,請(qǐng)查收!');
完整源碼:短信發(fā)送工廠方法 說(shuō)明
是不是很清晰了? 沒(méi)錯(cuò),我們有兩個(gè)產(chǎn)品,一個(gè)是Message,一個(gè)是Push,分別是發(fā)信息和發(fā)推送 抽象工廠只是要求我們的接口實(shí)現(xiàn)者必須去實(shí)現(xiàn)兩個(gè)方法,返回發(fā)短信和發(fā)推送的對(duì)象 你說(shuō)我只想發(fā)短信不想發(fā)推送可以嗎?當(dāng)然可以啦,不去調(diào)用createPush()方法不就行了 抽象工廠最適合什么場(chǎng)景?很明顯,一系列相關(guān)對(duì)象的創(chuàng)建 工廠方法模式是抽象工廠的核心,相當(dāng)于多個(gè)工廠方法被放到一個(gè)大工廠中生產(chǎn)一整套產(chǎn)品(包含周邊)而不是一件單獨(dú)的產(chǎn)品
下期看點(diǎn)有沒(méi)有化過(guò)妝?有沒(méi)有搭配過(guò)衣服?化妝我們要一層一層的化,衣服我們要從里向外的穿?都沒(méi)試過(guò)的話(海南程序員全年背心+短褲嗎???那你也得穿內(nèi)褲吧!?。?...沒(méi)關(guān)系,先帶你了解下裝飾者模式。 Github原稿:https://github.com/zhangyue0503/designpatterns-php/blob/master/03.abstract-factory/blog.md
|