適配器模式
一、什麼是適配器模式
適配器模式(Adapter Pattern)是作爲兩個不兼容的接口之間的橋樑。這種類型的設計模式屬於結構型模式,它結合了兩個獨立接口的功能。將一個類的接口轉換成客戶希望的另外一個接口。適配器模式使得原本由於接口不兼容而不能一起工作的那些類可以一起工作。
主要解決在軟件系統中,常常要將一些"現存的對象"放到新的環境中,而新環境要求的接口是現對象不能滿足的。
二、使用條件
1、系統需要使用現有的類,而此類的接口不符合系統的需要。
2、想要建立一個可以重複使用的類,用於與一些彼此之間沒有太大關聯的一些類,包括一些可能在將來引進的類一起工作,這些源類不一定有一致的接口。
3、通過接口轉換,將一個類插入另一個類系中。(比如老虎和飛禽,現在多了一個飛虎,在不增加實體的需求下,增加一個適配器,在裏面包容一個虎對象,實現飛的接口。)
三、優點和缺點
優點:
- 可以讓任何兩個沒有關聯的類一起運行。
- 提高了類的複用。
- 增加了類的透明度。
- 靈活性好。
缺點:
- 過多地使用適配器,會讓系統非常零亂,不易整體進行把握。比如,明明看到調用的是 A 接口,其實內部被適配成了 B 接口的實現,一個系統如果太多出現這種情況,無異於一場災難。因此如果不是很有必要,可以不使用適配器,而是直接對系統進行重構。
- 由於 JAVA 至多繼承一個類,所以至多隻能適配一個適配者類,而且目標類必須是抽象類。
注意事項:適配器不是在詳細設計時添加的,而是解決正在服役的項目的問題。
三、實現方法
1、結構圖
2、主要角色
配器模式包含三個角色:
Target(目標抽象類):目標抽象類定義客戶所需的接口,可以是一個抽象類或接口,也可以是具體類。
Adapter(適配器類):它可以調用另一個接口,作爲一個轉換器,對Adaptee和Target進行適配。它是適配器模式的核心。
Adaptee(適配者類):適配者即被適配的角色,它定義了一個已經存在的接口,這個接口需要適配,適配者類包好了客戶希望的業務方法。
3、java 代碼實現
目標接口 UserInfo
package com.designpattern.adapterPattern;
/**
* 通過不同的搜索類型獲取用戶信息
*
* @author zhongtao on 2018/12/4
*/
public interface UserInfo {
void getInfoByType(String type, String search);
}
適配者接口 UserDetailInfo
package com.designpattern.adapterPattern;
/**
* 用戶詳情 兩個接口 通過id或phone獲取用戶信息
*
* @author zhongtao on 2018/12/4
*/
public interface UserDetailInfo {
void getInfoById(String id);
void getInfoByPhone(String phone);
}
實現適配者接口
/**
* 實現通過id獲取用戶信息
*
* @author zhongtao on 2018/12/4
*/
public class UserInfoById implements UserDetailInfo {
@Override
public void getInfoById(String id) {
System.out.println("通過用戶id獲取用戶信息,id號:" + id);
}
@Override
public void getInfoByPhone(String phone) {
}
}
/**
* 實現通過電話號碼獲取用戶信息
*
* @author zhongtao on 2018/12/4
*/
public class UserInfoByPhone implements UserDetailInfo {
@Override
public void getInfoById(String id) {
}
@Override
public void getInfoByPhone(String phone) {
System.out.println("通過電話號碼獲取用戶信息,電話號碼:" + phone);
}
}
適配器類
package com.designpattern.adapterPattern;
/**
* 獲取用戶類型的適配器
*
* @author zhongtao on 2018/12/4
*/
public class UserInfoAdapter implements UserInfo {
UserDetailInfo userDetailInfo;
public UserInfoAdapter(String type) {
if (type.equals("ID")) {
userDetailInfo = new UserInfoById();
} else if (type.equals("PHONE")) {
userDetailInfo = new UserInfoByPhone();
}
}
@Override
public void getInfoByType(String type, String search) {
if (type.equals("ID")) {
userDetailInfo.getInfoById(search);
} else if (type.equals("PHONE")) {
userDetailInfo.getInfoByPhone(search);
}
}
}
適配目標實體類
package com.designpattern.adapterPattern;
/**
* 客戶需要的目標類
*
* @author zhongtao on 2018/12/4
*/
public class UserInfoSearch implements UserInfo {
/**
* 適配器
*/
UserInfoAdapter userInfoAdapter;
@Override
public void getInfoByType(String type, String search) {
if (type.equals("NAME")) {
System.out.println("通過用戶名獲取用戶信息,用戶名:" + search);
} else if (type.equals("ID") || type.equals("PHONE")) {
userInfoAdapter = new UserInfoAdapter(type);
userInfoAdapter.getInfoByType(type, search);
} else {
System.out.println("無效的搜索類型,searchType:" + type);
}
}
}
測試適配者模式
package com.designpattern.adapterPattern;
import org.junit.Test;
/**
* 測試適配器模式
*
* @author zhongtao on 2018/12/4
*/
public class UserInfoSearchTest {
/**
* 測試適配器模式
*/
@Test
public void testAdapter() {
UserInfoSearch userInfoSearch = new UserInfoSearch();
userInfoSearch.getInfoByType("ID", "123456");
userInfoSearch.getInfoByType("PHONE", "13433202125");
userInfoSearch.getInfoByType("NAME", "Peter");
userInfoSearch.getInfoByType("EMAIL", "[email protected]");
}
}
測試結果
通過用戶id獲取用戶信息,id號:123456
通過電話號碼獲取用戶信息,電話號碼:13433202125
通過用戶名獲取用戶信息,用戶名:Peter
無效的搜索類型,searchType:EMAIL