溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

講解PHP設(shè)計(jì)模式適配器模式Adapter-結(jié)構(gòu)型

發(fā)布時(shí)間:2021-03-08 15:54:22 來(lái)源:億速云 閱讀:171 作者:TREX 欄目:開發(fā)技術(shù)

本篇內(nèi)容介紹了“講解PHP設(shè)計(jì)模式適配器模式Adapter-結(jié)構(gòu)型”的有關(guān)知識(shí),在實(shí)際案例的操作過(guò)程中,不少人都會(huì)遇到這樣的困境,接下來(lái)就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

1. 概述:

         接口的改變,是一個(gè)需要程序員們必須(雖然很不情愿)接受和處理的普遍問(wèn)題。程序提供者們修改他們的代碼;系統(tǒng)庫(kù)被修正;各種程序語(yǔ)言以及相關(guān)庫(kù)的發(fā)展和進(jìn)化。

        例子1:iphone4,你即可以使用UBS接口連接電腦來(lái)充電,假如只有iphone沒(méi)有電腦,怎么辦呢?蘋果提供了iphone電源適配器??梢允褂眠@個(gè)電源適配器充電。這個(gè)iphone的電源適配器就是類似我們說(shuō)的適配器模式。(電源適配器就是把電源變成需要的電壓,也就是適配器的作用是使得一個(gè)東西適合另外一個(gè)東西。)

       例子2:最典型的例子就是很多功能手機(jī),每一種機(jī)型都自帶有從電器,有一天自帶充電器壞了,而且市場(chǎng)沒(méi)有這類型充電器可買了。怎么辦?萬(wàn)能充電器就可以解決。這個(gè)萬(wàn)能充電器就是適配器。

2. 問(wèn)題

     你如何避免因外部庫(kù)的API改變而帶來(lái)的不便?假如你寫了一個(gè)庫(kù),你能否提供一種方法允許你軟件的現(xiàn)有用戶進(jìn)行完美地升級(jí),即使你已經(jīng)改變了你的API?為了更好地適宜于你的需要,你應(yīng)該如何改變一個(gè)對(duì)象的接口?

3. 解決方案

        適配器(Adapter)模式為對(duì)象提供了一種完全不同的接口。你可以運(yùn)用適配器(Adapter)來(lái)實(shí)現(xiàn)一個(gè)不同的類的常見接口,同時(shí)避免了因升級(jí)和拆解客戶代碼所引起的糾紛。

    適配器模式(Adapter Pattern),把一個(gè)類的接口變換成客戶端所期待的另一種接口, Adapter模式使原本因接口不匹配(或者不兼容)而無(wú)法在一起工作的兩個(gè)類能夠在一起工作。又稱為轉(zhuǎn)換器模式、變壓器模式、包裝(Wrapper)器模式(把已有的一些類包裝起來(lái),使之能有滿足需要的接口)。
     考慮一下當(dāng)(不是假設(shè)!)一個(gè)第三方庫(kù)的API改變將會(huì)發(fā)生什么。過(guò)去你只能是咬緊牙關(guān)修改所有的客戶代碼,而情況往往還不那么簡(jiǎn)單。你可能正從事一項(xiàng)新的項(xiàng)目,它要用到新版本的庫(kù)所帶來(lái)的特性,但你已經(jīng)擁有許多舊的應(yīng)用程序,并且它們與以前舊版本的庫(kù)交互運(yùn)行地很好。你將無(wú)法證明這些新特性的利用價(jià)值,如果這次升級(jí)意味著將要涉及到其它應(yīng)用程序的客戶代碼。

4. 分類

共有兩類適配器模式:1.類的適配器模式(采用繼承實(shí)現(xiàn))2.對(duì)象適配器(采用對(duì)象組合方式實(shí)現(xiàn))

1)類適配器模式    ——適配器繼承自已實(shí)現(xiàn)的類(一般多重繼承)。

Adapter與Adaptee是繼承關(guān)系

1、用一個(gè)具體的Adapter類和Target進(jìn)行匹配。結(jié)果是當(dāng)我們想要一個(gè)匹配一個(gè)類以及所有它的子類時(shí),類Adapter將不能勝任工作
2、使得Adapter可以重定義Adaptee的部分行為,因?yàn)锳dapter是Adaptee的一個(gè)子集
3、僅僅引入一個(gè)對(duì)象,并不需要額外的指針以間接取得adaptee
2)對(duì)象適配器模式—— 適配器容納一個(gè)它包裹的類的實(shí)例。在這種情況下,適配器調(diào)用被包裹對(duì)象的物理實(shí)體。

Adapter與Adaptee是委托關(guān)系

1、允許一個(gè)Adapter與多個(gè)Adaptee同時(shí)工作。Adapter也可以一次給所有的Adaptee添加功能
2、使用重定義Adaptee的行為比較困難
無(wú)論哪種適配器,它的宗旨都是:保留現(xiàn)有類所提供的服務(wù),向客戶提供接口,以滿足客戶的期望。
即在不改變?cè)邢到y(tǒng)的基礎(chǔ)上,提供新的接口服務(wù)。

5. 適用性

以下情況使用Adapter模式:

1 ? 你想使用一個(gè)已經(jīng)存在的類,而它的接口不符合你的需求。
2 ? 你想創(chuàng)建一個(gè)可以復(fù)用的類,該類可以與其他不相關(guān)的類或不可預(yù)見的類(即那些接口可能不一定兼容的類)協(xié)同工作。
3 ?(僅適用于對(duì)象Adapter)你想使用一些已經(jīng)存在的子類,但是不可能對(duì)每一個(gè)都進(jìn)行子類化以匹配它們的接口。對(duì)象適配器可以適配它的父類接口。即僅僅引入一個(gè)對(duì)象,并不需要額外的指針以間接取得adaptee。

6. 結(jié)構(gòu)

類適配器使用多重繼承對(duì)一個(gè)接口與另一個(gè)接口進(jìn)行匹配,如下圖所示:

講解PHP設(shè)計(jì)模式適配器模式Adapter-結(jié)構(gòu)型

對(duì)象匹配器依賴于對(duì)象組合,如下圖所示:

講解PHP設(shè)計(jì)模式適配器模式Adapter-結(jié)構(gòu)型

7. 構(gòu)建模式的組成

?目標(biāo)角色(Target):— 定義Client使用的與特定領(lǐng)域相關(guān)的接口。
? 客戶角色(Client):與符合Target接口的對(duì)象協(xié)同。
? 被適配橘色(Adaptee):定義一個(gè)已經(jīng)存在并已經(jīng)使用的接口,這個(gè)接口需要適配。
? 適配器角色(Adapte) :適配器模式的核心。它將對(duì)被適配Adaptee角色已有的接口轉(zhuǎn)換為目標(biāo)角色Target匹配的接口。對(duì)Adaptee的接口與Target接口進(jìn)行適配.

8. 效果

類適配器和對(duì)象適配器有不同的權(quán)衡。

類適配器

? 用一個(gè)具體的Adapter類對(duì)Adaptee和Target進(jìn)行匹配。結(jié)果是當(dāng)我們想要匹配一個(gè)類以及所有它的子類時(shí),類Adapter將不能勝任工作。
? 使得Adapter可以重定義Adaptee的部分行為,因?yàn)锳dapter是Adaptee的一個(gè)子類。
? 僅僅引入了一個(gè)對(duì)象,并不需要額外的指針以間接得到 Adaptee。

對(duì)象適配器則

? 允許一個(gè)Adapter與多個(gè)Adaptee—即Adaptee本身以及它的所有子類(如果有子類的話)—同時(shí)工作。Adapter也可以一次給所有的Adaptee添加功能。
? 使得重定義Adaptee的行為比較困難。這就需要生成Adaptee的子類并且使得Adapter引用這個(gè)子類而不是引用Adaptee本身。

使用Adapter模式時(shí)需要考慮的其他一些因素有:

1) Adapter的匹配程度 對(duì)Adaptee的接口與Target的接口進(jìn)行匹配的工作量各個(gè)Adapter可能不一樣。工作范圍可能是,從簡(jiǎn)單的接口轉(zhuǎn)換(例如改變操作名 )到支持完全不同的操作集合。Adapter的工作量取決于Target接口與Adaptee接口的相似程度
2) 可插入的Adapter   當(dāng)其他的類使用一個(gè)類時(shí),如果所需的假定條件越少,這個(gè)類就更具可復(fù)用性。如果將接口匹配構(gòu)建為一個(gè)類,
就不需要假定對(duì)其他的類可見的是一個(gè)相同的接口。也就是說(shuō),接口匹配使得我們可以將自己的類加入到一些現(xiàn)有的系統(tǒng)中去,
而這些系統(tǒng)對(duì)這個(gè)類的接口可能會(huì)有所不同。 
3) 使用雙向適配器提供透明操作 使用適配器的一個(gè)潛在問(wèn)題是,它們不對(duì)所有的客戶都透明。被適配的對(duì)象不再兼容 Adaptee的接口,
因此并不是所有 Adaptee對(duì)象可以被使用的地方它都可以被使用。雙向適配器提供了這樣的透明性。
在兩個(gè)不同的客戶需要用不同的方式查看同一個(gè)對(duì)象時(shí),雙向適配器尤其有用。

9. 實(shí)現(xiàn)

類適配器使用的是繼承

讓我們看看當(dāng)API改變時(shí),如何保護(hù)應(yīng)用程序不受影響。

<?php
/**
 * 類適配器模式
 * @author guisu
 * 
 */
 
/**
 * 目標(biāo)角色
 * @version 1.0
 */
class Target {
 
  /**
   * 這個(gè)方法將來(lái)有可能改進(jìn)
   */
  public function hello(){
   echo 'Hello ';
  }
 
  /**
   * 目標(biāo)點(diǎn)
   */
  public function world(){
   echo 'world';
  }
}
 
/**
 * Client 程序
 *
 */
class Client {
 
  /**
   * Main program.
   */
  public static function main() {
    $Target = new Target();
    $Target->hello();
    $Target->world();
 
  }
 
}
Client::main();
?>

我們Target已經(jīng)明確指出hello()方法會(huì)在未來(lái)的版本中改進(jìn),甚至不被支持或者淘汰。接下來(lái),現(xiàn)在假設(shè)第二版的Target已經(jīng)發(fā)布。一個(gè)全新的greet()方法代替了hello()。

<?php
/**
 * 類適配器模式
 * @author guisu
 * 
 */
 
/**
 * 目標(biāo)角色
 * @version 2.0
 */
class Target {
 
  /**
   * 這個(gè)方法將來(lái)有可能繼續(xù)改進(jìn)
   */
  public function greet(){
   echo 'Greet ';
  }
 
  /**
   * 目標(biāo)點(diǎn)
   */
  public function world(){
   echo 'world';
  }
}

如果我們繼續(xù)使用原來(lái)的client代碼,肯定會(huì)報(bào)錯(cuò),找不到hello方法。

針對(duì)API“升級(jí)”的解決辦法就是創(chuàng)建一個(gè)適配器(Adapter)。

類適配器使用的是繼承:

<?php
/**
 * 類適配器模式
 * @author guisu
 * 
 */
 
/**
 * 目標(biāo)角色
 * @version 2.0
 */
interface Target {
 
  /**
   * 源類的方法:這個(gè)方法將來(lái)有可能繼續(xù)改進(jìn)
   */
  public function hello();
 
  /**
   * 目標(biāo)點(diǎn)
   */
  public function world();
}
 
/**
 * 源角色:被適配的角色
 */
class Adaptee {
 /**
   * 源類含有的方法
   */
  public function world() {
    echo ' world <br />';
  }
 
  /**
   * 加入新的方法
   */
  public function greet() {
    echo ' Greet ';
  }
}
 
/**
 * 類適配器角色
 */
class Adapter extends Adaptee implements Target {
 
  /**
   * 源類中沒(méi)有world方法,在此補(bǔ)充
   */
  public function hello() {
    parent::greet();
  }
 
}
/**
 * 客戶端程序
 *
 */
class Client {
 
  /**
   * Main program.
   */
  public static function main() {
    $adapter = new Adapter();
    $adapter->hello();
    $adapter->world();
  }
}
Client::main();
?>

對(duì)象適配器使用的是委派

<?php
/**
 * 類適配器模式
 * @author guisu
 * 
 */
 
/**
 * 目標(biāo)角色
 * @version 2.0
 */
interface Target {
 
  /**
   * 源類的方法:這個(gè)方法將來(lái)有可能繼續(xù)改進(jìn)
   */
  public function hello();
 
  /**
   * 目標(biāo)點(diǎn)
   */
  public function world();
}
 
/**
 * 源角色:被適配的角色
 */
class Adaptee {
 /**
   * 源類含有的方法
   */
  public function world() {
    echo ' world <br />';
  }
 
  /**
   * 加入新的方法
   */
  public function greet() {
    echo ' Greet ';
  }
}
 
/**
 * 類適配器角色
 */
class Adapter implements Target {
 
 private $_adaptee;
 /**
  * construct
  *
  * @param Adaptee $adaptee
  */
  public function __construct(Adaptee $adaptee) {
    $this->_adaptee = $adaptee;
  }
 
  /**
   * 源類中沒(méi)有world方法,在此補(bǔ)充
   */
  public function hello() {
    $this->_adaptee->greet();
  }
 
  /**
   * 源類中沒(méi)有world方法,在此補(bǔ)充
   */
  public function world() {
    $this->_adaptee->world();
  }
}
/**
 * 客戶端程序
 *
 */
class Client {
 
  /**
   * Main program.
   */
  public static function main() {
   $adaptee = new Adaptee();
    $adapter = new Adapter($adaptee);
    $adapter->hello();
    $adapter->world();
  }
}
Client::main();
?>

如例中代碼所示,你可以運(yùn)用適配器(Adapter)模式來(lái)避免因外部庫(kù)改變所帶來(lái)的不便——倘若向上兼容。作為某個(gè)庫(kù)的開發(fā)者,你應(yīng)該獨(dú)立編寫適配器,使你的用戶更簡(jiǎn)便地使用新版本的庫(kù),而不用去修改他們現(xiàn)有的全部代碼。

     GoF書中提出的適配器(Adapter)模式更傾向于運(yùn)用繼承而不是組成。這在強(qiáng)類型語(yǔ)言中是有利的,因?yàn)檫m配器(Adapter)事實(shí)上是一個(gè)目標(biāo)類的子類,因而能更好地與類中方法相結(jié)合。

了更好的靈活性,我個(gè)人比較傾向于組成的方法(特別是在結(jié)合了依賴性倒置的情況下);盡管如此,繼承的方法提供兩種版本的接口,或許在你的實(shí)際運(yùn)用中反而是一個(gè)提高靈活性的關(guān)鍵。

10.適配器模式與其它相關(guān)模式

橋梁模式(bridge模式):橋梁模式與對(duì)象適配器類似,但是橋梁模式的出發(fā)點(diǎn)不同:橋梁模式目的是將接口部分和實(shí)現(xiàn)部分分離,從而對(duì)它們可以較為容易也相對(duì)獨(dú)立的加以改變。而對(duì)象適配器模式則意味著改變一個(gè)已有對(duì)象的接口

裝飾器模式(decorator模式):裝飾模式增強(qiáng)了其他對(duì)象的功能而同時(shí)又不改變它的接口。因此裝飾模式對(duì)應(yīng)用的透明性比適配器更好。結(jié)果是decorator模式支持遞歸組合,而純粹使用適配器是不可能實(shí)現(xiàn)這一點(diǎn)的。

Facade(外觀模式):適配器模式的重點(diǎn)是改變一個(gè)單獨(dú)類的API。Facade的目的是給由許多對(duì)象構(gòu)成的整個(gè)子系統(tǒng),提供更為簡(jiǎn)潔的接口。而適配器模式就是封裝一個(gè)單獨(dú)類,適配器模式經(jīng)常用在需要第三方API協(xié)同工作的場(chǎng)合,設(shè)法把你的代碼與第三方庫(kù)隔離開來(lái)。

適配器模式與外觀模式都是對(duì)現(xiàn)相存系統(tǒng)的封裝。但這兩種模式的意圖完全不同,前者使現(xiàn)存系統(tǒng)與正在設(shè)計(jì)的系統(tǒng)協(xié)同工作而后者則為現(xiàn)存系統(tǒng)提供一個(gè)更為方便的訪問(wèn)接口。簡(jiǎn)單地說(shuō),適配器模式為事后設(shè)計(jì),而外觀模式則必須事前設(shè)計(jì),因?yàn)橄到y(tǒng)依靠于外觀。總之,適配器模式?jīng)]有引入新的接口,而外觀模式則定義了一個(gè)全新的接口。

代理模式(Proxy )在不改變它的接口的條件下,為另一個(gè)對(duì)象定義了一個(gè)代理。

裝飾者模式,適配器模式,外觀模式三者之間的區(qū)別:

裝飾者模式的話,它并不會(huì)改變接口,而是將一個(gè)一個(gè)的接口進(jìn)行裝飾,也就是添加新的功能。

適配器模式是將一個(gè)接口通過(guò)適配來(lái)間接轉(zhuǎn)換為另一個(gè)接口。

外觀模式的話,其主要是提供一個(gè)整潔的一致的接口給客戶端。

“講解PHP設(shè)計(jì)模式適配器模式Adapter-結(jié)構(gòu)型”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI