1. 程式人生 > 實用技巧 >PHP設計模式之抽象工廠模式

PHP設計模式之抽象工廠模式

工廠模式系列中的重頭戲來了,沒錯,那正是傳聞中的抽象工廠模式。初次聽到這個名字的時候你有什麼感覺?反正我是感覺這貨應該是非常高大上的,畢竟包含著“抽象”兩個字。話說這兩個字在開發中真的是有點高大上的感覺,一帶上抽象兩字就好像哪哪都很厲害了呢。不過,抽象工廠也確實可以說是工廠模式的大哥大。

Gof類圖及解釋

其實只要理解了工廠方法模式,就很容易明白抽象工廠模式。怎麼說呢?還是一樣的延遲到子類,還是一樣的返回指定的物件。只是抽象工廠裡面不僅僅只返回一個物件,而是返回一堆。

GoF定義:提供一個建立一系列相關或相互依賴物件的介面,而無需指定它們具體的類。

GoF類圖

  • 左邊是兩個工廠1和2,都繼承一個抽象工廠,都實現了CreateProductA和CreateProductB方法
  • 工廠1生產的是ProductA1和ProductB1
  • 同樣的,工廠2生產的是ProductA2和ProductB2

程式碼實現

// 商品A抽象介面
interface AbstractProductA
{
    public function show(): void;
}

// 商品A1實現
class ProductA1 implements AbstractProductA
{
    public function show(): void
    {
        echo 'ProductA1 is Show!' . PHP_EOL;
    }
}
// 商品A2實現
class ProductA2 implements AbstractProductA
{
    public function show(): void
    {
        echo 'ProductA2 is Show!' . PHP_EOL;
    }
}

// 商品B抽象介面
interface AbstractProductB
{
    public function show(): void;
}
// 商品B1實現
class ProductB1 implements AbstractProductB
{
    public function show(): void
    {
        echo 'ProductB1 is Show!' . PHP_EOL;
    }
}
// 商品B2實現
class ProductB2 implements AbstractProductB
{
    public function show(): void
    {
        echo 'ProductB2 is Show!' . PHP_EOL;
    }
}

商品的實現,東西很多吧,這回其實是有四件商品了分別是A1、A2、B1和B2,他們之間假設有這樣的關係,A1和B1是同類相關的商品,B1和B2是同類相關的商品

// 抽象工廠介面
interface AbstractFactory
{
    // 建立商品A
    public function CreateProductA(): AbstractProductA;
    // 建立商品B
    public function CreateProductB(): AbstractProductB;
}

// 工廠1,實現商品A1和商品B1
class ConcreteFactory1 implements AbstractFactory
{
    public function CreateProductA(): AbstractProductA
    {
        return new ProductA1();
    }
    public function CreateProductB(): AbstractProductB
    {
        return new ProductB1();
    }
}

// 工廠2,實現商品A2和商品B2
class ConcreteFactory2 implements AbstractFactory
{
    public function CreateProductA(): AbstractProductA
    {
        return new ProductA2();
    }
    public function CreateProductB(): AbstractProductB
    {
        return new ProductB2();
    }
}

而我們的工廠也是工廠1和工廠2,工廠1生產的是A1和B1這兩種相關聯的產品,工廠2生產的是A2和B2這兩種商品。好吧,我知道這裡還是有點抽象,可能還是搞不懂為什麼要這樣,我們繼續以手機生產來舉例。

我們的手機品牌起來了,所以周邊如手機膜、手機殼也交給了富X康(AbstractFactory)來幫我搞定。上回說到,我已經有幾款不同型別的手機了,於是還是按原來那樣,衡陽工廠(Factory1)生產型號1001的手機(ProductA1),同時型號1001手機的手機膜(ProductB1)和手機殼(ProductC1)也是衡陽工廠生產出來。而型號1002的手機(ProductA2)還是在鄭州工廠(Factory2),這個型號的手機膜(ProductB2)和手機膜(ProductC2)也就交給他們來搞定吧。於是,我還是隻去跟總廠下單,他們讓不同的工廠給我生產了一整套的手機產品,可以直接賣套裝咯!!

完整程式碼:抽象工廠模式

例項

是不是看得還是有點暈。其實說簡單點,真的就是在一個工廠類中通過不同的方法返回不同的物件而已。讓我們再次用發簡訊的例項來講解吧!

場景:這次我們有個業務需求是,不僅要發簡訊,還要同時發一條推送。簡訊的目的是通知使用者有新的活動參加,而推送不僅通知有新的活動,直接點選就可以進去領紅包了,是不是很興奮。還好之前我們的選擇的雲服務供應商都是即有簡訊也有推送介面的,所以我們就直接用抽象工廠來實現吧!

簡訊傳送類圖

<?php

interface Message {
    public function send(string $msg);
}

class AliYunMessage implements Message{
    public function send(string $msg){
        // 呼叫介面,傳送簡訊
        // xxxxx
        return '阿里雲簡訊(原阿里大魚)傳送成功!簡訊內容:' . $msg;
    }
}

class BaiduYunMessage implements Message{
    public function send(string $msg){
        // 呼叫介面,傳送簡訊
        // xxxxx
        return '百度SMS簡訊傳送成功!簡訊內容:' . $msg;
    }
}

class JiguangMessage implements Message{
    public function send(string $msg){
        // 呼叫介面,傳送簡訊
        // xxxxx
        return '極光簡訊傳送成功!簡訊內容:' . $msg;
    }
}

interface Push {
    public function send(string $msg);
}

class AliYunPush implements Push{
    public function send(string $msg){
        // 呼叫介面,傳送客戶端推送
        // xxxxx
        return '阿里雲Android&iOS推送傳送成功!推送內容:' . $msg;
    }
}

class BaiduYunPush implements Push{
    public function send(string $msg){
        // 呼叫介面,傳送客戶端推送
        // xxxxx
        return '百度Android&iOS雲推送傳送成功!推送內容:' . $msg;
    }
}

class JiguangPush implements Push{
    public function send(string $msg){
        // 呼叫介面,傳送客戶端推送
        // xxxxx
        return '極光推送傳送成功!推送內容:' . $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();
    }
}

// 當前業務需要使用阿里雲
$factory = new AliYunFactory();
// $factory = new BaiduYunFactory();
// $factory = new JiguangFactory();
$message = $factory->createMessage();
$push = $factory->createPush();
echo $message->send('您已經很久沒有登入過系統了,記得回來哦!');
echo $push->send('您有新的紅包已到帳,請查收!');

完整原始碼:簡訊傳送工廠方法

說明

  • 是不是很清晰了?
  • 沒錯,我們有兩個產品,一個是Message,一個是Push,分別是發信息和發推送
  • 抽象工廠只是要求我們的介面實現者必須去實現兩個方法,返回發簡訊和發推送的物件
  • 你說我只想發簡訊不想發推送可以嗎?當然可以啦,不去呼叫createPush()方法不就行了
  • 抽象工廠最適合什麼場景?很明顯,一系列相關物件的建立
  • 工廠方法模式是抽象工廠的核心,相當於多個工廠方法被放到一個大工廠中生產一整套產品(包含周邊)而不是一件單獨的產品

下期看點

有沒有化過妝?有沒有搭配過衣服?化妝我們要一層一層的化,衣服我們要從裡向外的穿?都沒試過的話(海南程式設計師全年背心+短褲嗎???那你也得穿內褲吧!!)....沒關係,先帶你瞭解下裝飾者模式

關注公眾號:【硬核專案經理】獲取最新文章

新增微信/QQ好友:【xiaoyuezigonggong/149844827】免費得PHP、專案管理學習資料

知乎、公眾號、抖音、頭條搜尋【硬核專案經理】

B站ID:482780532