適配器模式,是一種結構型設計模式,又稱轉換器模式。它把一個類的接口變換成客戶端所需要的另一種接口,從而使原來因為接口不匹配而無法一起工作的兩個類能夠一起工作。顧名思義,它是進行適應與匹配工作的功能模塊。當一個對象或類的接口不能匹配用戶所需要的接口時,適配器就充當中間轉換的角色,以達到兼容用戶接口的目的,同時適配器也實現(xiàn)了客戶端與接口的解耦,提高了組件的可復用性。
在實際的軟件系統(tǒng)設計和開發(fā)中,為了完成某項工作需要購買一個第三方的庫來加快開發(fā)。這帶來一個問題,在應用程序中已經設計好的功能接口,與這個第三方提供的接口不一致。為了使得這些接口不兼容的類可以在一起工作,適配器模式提供了一種接口的適配機制。
適配器模式的設計思想在生活中經常會應用到,如我們在給手機充電的時候,不可能直接在220V電源上直接充電,而是用手機充電器轉換成手機需要的電壓才可以正常充電,否則就不可以完成充電,這個充電器就起到了適配的作用。
適配器模式簡介
定義
適配器模式是通過一個類的接口轉換成客戶希望的另外一個接口,使原本由于接口不兼容而不能一起工作的那些類可以一起工作。
適配器從結構上可以分為類適配器和對象適配器。其中類適配器使用繼承關系來對類進行適配,而對象適配器是使用對象引用的方法來進行適配的。
這種設計模式主要使用了依賴倒置原則和開閉原則:
依賴倒置原則的使用:使用對象組合的方式,以目標接口包裝被適配者,這使被適配者的任何子類都可以被適配器使用,提高了軟件結構的復用性。適配器模式將被適配者和目標接口綁定在一起,而不是和實現(xiàn)進行綁定,這有利于系統(tǒng)擴展。開閉原則的使用:將原有的接口轉化成另一種接口,既不影響原有系統(tǒng)的運行,又能高效地擴展新功能。適配器模式使用的場景:
軟件系統(tǒng)結構需要升級或者擴展,又不想影響原有系統(tǒng)的穩(wěn)定運行。轉換類之間的差異不是很大。想創(chuàng)建一個可以復用的類,該類可以與其他不相關的類或不可預見的類協(xié)同工作。眾所周知,反復修改代碼的代價是巨大的,因為所有依賴關系都要受到牽連,這不但會引入更多沒有必要的開發(fā)與測試工作,而且可能會帶來不可預知的風險,結果得不償失。適配器模式讓兼容性問題,在不必修改任何代碼的情況下得以解決,其中適配器類是核心。
適配器模式的UML類圖
對象適配器模式的各角色定義如下:
Target(目標接口):客戶端要使用的目標接口標準,也就是封裝被適配者的接口。Adapter(適配器):實現(xiàn)了目標接口,負責將被適配者的接口specificRequest()適配(轉換)為目標接口request()。Adaptee(被適配者):被適配者的接口標準,不能兼容目標接口的問題接口,可以有多種實現(xiàn)類。Client(客戶端):目標接口的使用者。2,舉例
#include
#include
// 定義適配者接口
struct adaptee {
void (*specificRequest)(void);
};
// 適配者操作
void adaptee_specificRequest(void)
{
printf("adaptee specific request\n");
}
// 初始化適配者
struct adaptee* adaptee_create(void)
{
struct adaptee* adaptee = (struct adaptee *)malloc(sizeof(struct adaptee *));
adaptee->specificRequest = adaptee_specificRequest;
return adaptee;
}
typedef struct adapter adapter;
// 定義目標接口
struct target {
void (*request)(struct adapter* this);
};
// 定義適配器
struct adapter {
struct adaptee* adaptee;
struct target target;
};
// 適配器請求操作
void adapter_request(struct adapter* this)
{
printf("adapter request\n");
this->adaptee->specificRequest();
}
// 初始化適配器
struct adapter* adapter_create(struct adaptee* adaptee)
{
struct adapter* adapter = (struct adapter *)malloc(sizeof(struct adapter *));
adapter->target.request = (void (*)())adapter_request;
adapter->adaptee = adaptee;
return adapter;
}
int main() {
// 創(chuàng)建適配者
struct adaptee* adaptee = adaptee_create();
// 創(chuàng)建適配器,并使用適配者初始化
struct adapter* adapter = adapter_create(adaptee);
// 調用適配器的請求方法
adapter->target.request(adapter);
// 釋放內存
free(adapter);
free(adaptee);
return 0;
}
如上例子中,定義了目標接口target,其中包含一個request方法。定義了適配者接口adaptee,其中包含一個specificRequest方法。在主函數中,創(chuàng)建了適配者并初始化。然后,創(chuàng)建了一個適配器adapter,并使用適配者對適配者進行初始化。適配器adapter實際上是一個結構體,其中包含一個request方法和一個指向被適配者的指針。
適配器的request方法實際上是調用適配器中的適配者的specificRequest方法。這樣,當我們調用適配器的request方法時,它會先執(zhí)行適配器自己的操作,然后再調用適配者的操作。最后,釋放內存以避免內存泄漏。
在實際應用中,適配器可以根據需要將適配者的方法轉換為目標接口的方法,并執(zhí)行其他操作。示例僅提供了一個簡單的框架,可以根據實際需求進行擴展和修改。





