PHP設(shè)計(jì)模式之適配器模式這個(gè)模式一直以來(lái)都有一個(gè)很經(jīng)典的例子,那就是插座!沒(méi)錯(cuò),當(dāng)我們從國(guó)外買(mǎi)回來(lái)電器,或者旅游出差去國(guó)外的時(shí)候,經(jīng)常會(huì)需要一個(gè)電源適配器,因?yàn)槲覈?guó)的電壓標(biāo)準(zhǔn)是220伏,而其他國(guó)家則有110伏的標(biāo)準(zhǔn)。而這個(gè)電源適配器正是適配器模式的一種標(biāo)志。當(dāng)對(duì)象不太符合要求的時(shí)候,給他加一個(gè)適配器唄?。?/p> Gof類(lèi)圖及解釋GoF定義:將一個(gè)類(lèi)的接口轉(zhuǎn)換成客戶(hù)希望的另外一個(gè)接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些類(lèi)可以一起工作 GoF類(lèi)圖
繼承式  組合式  代碼實(shí)現(xiàn)
interface Target{ function Request() : void; }
定義一個(gè)接口契約,也可以是一個(gè)正常的有實(shí)現(xiàn)方法的類(lèi)(后面的例子我們會(huì)用類(lèi)) class Adapter implements Target{ private $adaptee;
function __constuct($adaptee){ $this->adaptee = $adaptee; }
function Request() : void { $this->adaptee->SpecificRequest(); } }
適配器實(shí)現(xiàn)這個(gè)接口契約,讓Request()方法得以實(shí)現(xiàn),但請(qǐng)注意,我們真正調(diào)用的其實(shí)是Adaptee類(lèi)中的方法 class Adaptee { function SpecificRequest() : void{ echo "I'm China Standard!"; } }
適配器有兩種形式,上方類(lèi)圖中給出了,我們代碼實(shí)現(xiàn)的組合形式的 繼承形式的在GoF書(shū)中是以C++為示例的,因?yàn)镃++可以實(shí)現(xiàn)多重繼承,但現(xiàn)在流行的大部分語(yǔ)言是以接口為形式的,也可以實(shí)現(xiàn),但使用這種形式的適配器不多 其實(shí)還是面向接口編程的一種思維,類(lèi)似于裝飾器對(duì)舊功能的包裝,我們這里就是直接去進(jìn)行了替換,但對(duì)外的調(diào)用還是保持不變 適配器模式其實(shí)很好理解,代碼真的就只有這么點(diǎn)
又說(shuō)到我的手機(jī)工廠了,這回咱們的生意真的做大了哦!賣(mài)到泰國(guó)、新加坡、印度尼西亞去了,反正有咖喱的地方都有我們的身影了。據(jù)說(shuō)是我們出了個(gè)咖喱色。換殼這事兒可不完全是因?yàn)槭艿街ZX亞的影響,而是真的經(jīng)過(guò)長(zhǎng)期的調(diào)研我們發(fā)現(xiàn)不同顏色在不同的地方銷(xiāo)量會(huì)更好。于是,富X康在原有的手機(jī)殼生產(chǎn)線(xiàn)(Target)上為我們加裝了一個(gè)噴涂適配器(adapter),當(dāng)我們需要其他顏色的殼時(shí),只需要這個(gè)適配器換不同的顏料就好啦(adaptee),直接裝上這個(gè)噴涂器,新的顏色的手機(jī)就誕生了。而當(dāng)向另外一個(gè)國(guó)家擴(kuò)展業(yè)務(wù)時(shí),我們換顏料就行啦,用太久了不行就連噴頭也換掉(是不是想起了連供打印機(jī)) 完整代碼:適配器模式 實(shí)例繼續(xù)發(fā)短信,看我能編到什么時(shí)候~~~ 各位大拿在對(duì)接信息、支付類(lèi)的接口時(shí),經(jīng)常會(huì)使用這些平臺(tái)提供的SDK。特別是有了Composer之后,安裝SDK就更加的方便了,但是,又有一個(gè)嚴(yán)重的問(wèn)題,這幫人做的SDK雖說(shuō)功能實(shí)現(xiàn)大同小異,但命名可是千差萬(wàn)別啊?。∥覀兊南到y(tǒng)原來(lái)一直使用的阿里云的業(yè)務(wù),但是這回要增加極光和百度云的信息功能,一來(lái)做個(gè)后備,二來(lái)根據(jù)不同業(yè)務(wù)使用不同的接口達(dá)到安全或節(jié)約的目的,有沒(méi)有辦法統(tǒng)一一下他們對(duì)外的接口,讓我們使用他們的SDK時(shí)能夠非常方便的和之前使用大家都已經(jīng)很習(xí)慣的阿里云的接口一樣呢?當(dāng)然有,給他們各自都上個(gè)適配器唄,實(shí)例化的時(shí)候大不了外面再套個(gè)工廠返回不同的適配器就好啦,只要適配器里的實(shí)現(xiàn)方法和阿里云一樣就OK啦! 短信發(fā)送類(lèi)圖
 完整源碼:短信發(fā)送適配器方法 <?php
class Message{ public function send(){ echo "阿里云發(fā)送短信!" . PHP_EOL; } public function push(){ echo "阿里云發(fā)送推送!" . 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 "極光發(fā)送短信!" . PHP_EOL; } public function push_msg(){ echo "極光發(fā)送推送!" . 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 "百度云發(fā)送短信!" . PHP_EOL; } public function transmission_push(){ echo "百度云發(fā)送推送!" . PHP_EOL; } }
$jiguangMessage = new JiguangMessage(); $baiduYunMessage = new BaiduYunMessage(); $message = new Message();
// 原來(lái)的老系統(tǒng)發(fā)短信,使用阿里云 $message->send(); $message->push();
// 部分模塊用極光發(fā)吧 $jgAdatper = new JiguangSDKAdapter($jiguangMessage); $jgAdatper->send(); $jgAdatper->push();
// 部分模塊用百度云發(fā)吧 $bdAatper = new BaiduYunSDKAdapter($baiduYunMessage); $bdAatper->send(); $bdAatper->push();
說(shuō)明
在這個(gè)例子中,我們有兩個(gè)適配器,因?yàn)橛袃蓚€(gè)SDK需要我們?nèi)ミm配,誰(shuí)說(shuō)只能有一個(gè)電源轉(zhuǎn)換器,萬(wàn)一哪個(gè)神奇的國(guó)度是用500伏的電壓呢,所以還是多帶個(gè)電源轉(zhuǎn)換器吧 這里我們是繼承的Message類(lèi),因?yàn)镸essage類(lèi)是之前已經(jīng)寫(xiě)好的代碼,里面可能有一些可以公用的方法,所以并沒(méi)有做接口抽象??梢钥紤]在重構(gòu)代碼的時(shí)候?qū)崿F(xiàn)提取一個(gè)抽象接口,但在這里只是為了演示適配器不一定只是能去針對(duì)接口,只要和原對(duì)象保持一致,不去繼承什么也是可以的,畢竟我們是弱類(lèi)型語(yǔ)言,如果是類(lèi)似于Java的強(qiáng)類(lèi)型,那么繼承或者實(shí)現(xiàn)還是很有必要的(多態(tài)性) 組合式的適配器與裝飾器類(lèi)似,都會(huì)維護(hù)一個(gè)外部對(duì)象,裝飾器更多的會(huì)使用原來(lái)的類(lèi)中的方法,對(duì)其進(jìn)行增加功能的操作,而適配器則很少去增加功能,而是直接替換掉 Laravel中的Filesystem模塊,有一個(gè)FilesystemAdapter類(lèi),我覺(jué)得沒(méi)啥可說(shuō)的了,很明顯的告訴大家咱用了適配器模式,好好研究一下吧 當(dāng)你想使用一個(gè)類(lèi),但他提供的內(nèi)容跟你的業(yè)務(wù)又不太匹配的時(shí)候;或者你想創(chuàng)建一個(gè)類(lèi),可以與其他不相關(guān)的類(lèi)或不可預(yù)見(jiàn)的類(lèi)協(xié)同工作的時(shí)候,不妨試試適配器模式吧
下期看點(diǎn)事件訂閱有沒(méi)有聽(tīng)說(shuō)過(guò)?沒(méi)有?如果地震算一個(gè)事件的話(huà),那么一旦發(fā)生這個(gè)災(zāi)難了,馬上會(huì)有種類(lèi)政府部門(mén)和社會(huì)團(tuán)隊(duì)開(kāi)始行動(dòng),救援、搶險(xiǎn)等各種工作馬上展開(kāi),我們可以把整個(gè)社會(huì)力量都當(dāng)做是訂閱者,包括我們每一個(gè)都會(huì)很關(guān)心災(zāi)區(qū)的情況。在這里,我們所有人都是觀察者。這下就很容易理解觀察者模式了吧,下節(jié)我們?cè)僭斒觯?/p>
|