天天看點

PHP設計模式之擴充卡模式PHP設計模式之擴充卡模式

PHP設計模式之擴充卡模式

這個模式一直以來都有一個很經典的例子,那就是插座!沒錯,當我們從國外買回來電器,或者旅遊出差去國外的時候,經常會需要一個電源擴充卡,因為我國的電壓标準是220伏,而其他國家則有110伏的标準。而這個電源擴充卡正是擴充卡模式的一種标志。當對象不太符合要求的時候,給他加一個擴充卡呗!!

Gof類圖及解釋

GoF定義:将一個類的接口轉換成客戶希望的另外一個接口。Adapter模式使得原本由于接口不相容而不能一起工作的那些類可以一起工作

GoF類圖

繼承式

PHP設計模式之擴充卡模式PHP設計模式之擴充卡模式

組合式

PHP設計模式之擴充卡模式PHP設計模式之擴充卡模式
代碼實作
interface Target{
    function Request() : void;
}
           

複制

定義一個接口契約,也可以是一個正常的有實作方法的類(後面的例子我們會用類)

class Adapter implements Target{
    private $adaptee;

    function __constuct($adaptee){
        $this->adaptee = $adaptee;
    }

    function Request() : void {
        $this->adaptee->SpecificRequest();
    }
}
           

複制

擴充卡實作這個接口契約,讓Request()方法得以實作,但請注意,我們真正調用的其實是Adaptee類中的方法

class Adaptee {
    function SpecificRequest() : void{
        echo "I'm China Standard!";
    }
}
           

複制

  • 擴充卡有兩種形式,上方類圖中給出了,我們代碼實作的組合形式的
  • 繼承形式的在GoF書中是以C++為示例的,因為C++可以實作多重繼承,但現在流行的大部分語言是以接口為形式的,也可以實作,但使用這種形式的擴充卡不多
  • 其實還是面向接口程式設計的一種思維,類似于裝飾器對舊功能的包裝,我們這裡就是直接去進行了替換,但對外的調用還是保持不變
  • 擴充卡模式其實很好了解,代碼真的就隻有這麼點

又說到我的手機工廠了,這回咱們的生意真的做大了哦!賣到泰國、新加坡、印度尼西亞去了,反正有咖喱的地方都有我們的身影了。據說是我們出了個咖喱色。換殼這事兒可不完全是因為受到諾X亞的影響,而是真的經過長期的調研我們發現不同顔色在不同的地方銷量會更好。于是,富X康在原有的手機殼生産線(Target)上為我們加裝了一個噴塗擴充卡(adapter),當我們需要其他顔色的殼時,隻需要這個擴充卡換不同的顔料就好啦(adaptee),直接裝上這個噴塗器,新的顔色的手機就誕生了。而當向另外一個國家擴充業務時,我們換顔料就行啦,用太久了不行就連噴頭也換掉(是不是想起了連供列印機)

完整代碼:擴充卡模式

執行個體

繼續發短信,看我能編到什麼時候~~~

各位大拿在對接資訊、支付類的接口時,經常會使用這些平台提供的SDK。特别是有了Composer之後,安裝SDK就更加的友善了,但是,又有一個嚴重的問題,這幫人做的SDK雖說功能實作大同小異,但命名可是千差萬别啊!!我們的系統原來一直使用的阿裡雲的業務,但是這回要增加極光和百度雲的資訊功能,一來做個後備,二來根據不同業務使用不同的接口達到安全或節約的目的,有沒有辦法統一一下他們對外的接口,讓我們使用他們的SDK時能夠非常友善的和之前使用大家都已經很習慣的阿裡雲的接口一樣呢?當然有,給他們各自都上個擴充卡呗,執行個體化的時候大不了外面再套個工廠傳回不同的擴充卡就好啦,隻要擴充卡裡的實作方法和阿裡雲一樣就OK啦!

短信發送類圖
PHP設計模式之擴充卡模式PHP設計模式之擴充卡模式

完整源碼:短信發送擴充卡方法

<?php

class Message{
    public function send(){
        echo "阿裡雲發送短信!" . PHP_EOL;
    }
    public function push(){
        echo "阿裡雲發送推送!" . PHP_EOL;
    }
}

class JiguangSDKAdapter extends Message{
    private $message;

    public function __construct($message){
        $this->message = $message;
    }

    public function send(){
        $this->message->send_out_msg();
    }
    public function push(){
        $this->message->push_msg();
    }
}

class JiguangMessage{
    public function send_out_msg(){
        echo "極光發送短信!" . PHP_EOL;
    }
    public function push_msg(){
        echo "極光發送推送!" . PHP_EOL;
    }
}
class BaiduYunSDKAdapter extends Message{
    private $message;

    public function __construct($message){
        $this->message = $message;
    }

    public function send(){
        $this->message->transmission_msg();
    }
    public function push(){
        $this->message->transmission_push();
    }
}
class BaiduYunMessage{
    public function transmission_msg(){
        echo "百度雲發送短信!" . PHP_EOL;
    }
    public function transmission_push(){
        echo "百度雲發送推送!" . PHP_EOL;
    }
}

$jiguangMessage = new JiguangMessage();
$baiduYunMessage = new BaiduYunMessage();
$message = new Message();

// 原來的老系統發短信,使用阿裡雲
$message->send();
$message->push();


// 部分子產品用極光發吧
$jgAdatper = new JiguangSDKAdapter($jiguangMessage);
$jgAdatper->send();
$jgAdatper->push();

// 部分子產品用百度雲發吧
$bdAatper = new BaiduYunSDKAdapter($baiduYunMessage);
$bdAatper->send();
$bdAatper->push();
           

複制

說明
  • 在這個例子中,我們有兩個擴充卡,因為有兩個SDK需要我們去适配,誰說隻能有一個電源轉換器,萬一哪個神奇的國度是用500伏的電壓呢,是以還是多帶個電源轉換器吧
  • 這裡我們是繼承的Message類,因為Message類是之前已經寫好的代碼,裡面可能有一些可以公用的方法,是以并沒有做接口抽象。可以考慮在重構代碼的時候實作提取一個抽象接口,但在這裡隻是為了示範擴充卡不一定隻是能去針對接口,隻要和原對象保持一緻,不去繼承什麼也是可以的,畢竟我們是弱類型語言,如果是類似于Java的強類型,那麼繼承或者實作還是很有必要的(多态性)
  • 組合式的擴充卡與裝飾器類似,都會維護一個外部對象,裝飾器更多的會使用原來的類中的方法,對其進行增加功能的操作,而擴充卡則很少去增加功能,而是直接替換掉
  • Laravel中的Filesystem子產品,有一個FilesystemAdapter類,我覺得沒啥可說的了,很明顯的告訴大家咱用了擴充卡模式,好好研究一下吧
  • 當你想使用一個類,但他提供的内容跟你的業務又不太比對的時候;或者你想建立一個類,可以與其他不相關的類或不可預見的類協同工作的時候,不妨試試擴充卡模式吧

下期看點

事件訂閱有沒有聽說過?沒有?如果地震算一個事件的話,那麼一旦發生這個災難了,馬上會有種類政府部門和社會團隊開始行動,救援、搶險等各種工作馬上展開,我們可以把整個社會力量都當做是訂閱者,包括我們每一個都會很關心災區的情況。在這裡,我們所有人都是觀察者。這下就很容易了解觀察者模式了吧,下節我們再詳述!