簡述
- 型別:結構型
- 目的:解決介面不兼容問題,
話不多說,看個案例吧,
優化案例
最初版v0
在真實的開發場景中,系統的每個模塊都是分配給不同的團隊或個人來開發的,這使得事前溝通變得尤為重要,且溝通問題也時有發生,現在公司有兩個模塊無法兼容,難道只能重寫其中的一個嗎?
class User {
String name;
String sex;
int age;
// 剩下的屬性就不寫了,都是廢話沒啥意義
public User(String name, String sex, int age) {
this.name = name;
this.sex = sex;
this.age = age;
}
}
interface Filter {
List<User> findAll();
User findByName(String name);
}
class FilterImpl implements Filter { // Adaptee
List<User> users;
public FilterImpl(List<User> users) {
this.users = users;
}
public List<User> findAll() {
return users;
}
public User findByName(String name) {
if (name == null) throw new RuntimeException("請輸入正確的ID!");
return (User) users.stream().filter(t -> name.equals(t.name)).findFirst().orElse(null);
}
}
interface JsonFilter { // Target
String allToJson();
String findByNameToJson(String id);
}
客戶想要查詢user并且回傳結果物件的Json,只是當前的兩個模塊沒法滿足需求,如果不想修改這兩個模塊,我們如何實作兩個模塊功能的整合呢?
修改版v1(類配接器)
我們引入一個新的類作為配接器來適配原有的兩個模塊,
class User {
String name;
String sex;
int age;
// 剩下的屬性就不寫了,都是廢話沒啥意義
public User(String name, String sex, int age) {
this.name = name;
this.sex = sex;
this.age = age;
}
}
interface Filter {
List<User> findAll();
User findByName(String name);
}
class FilterImpl implements Filter { // Adaptee
List<User> users;
public FilterImpl(List<User> users) {
this.users = users;
}
public List<User> findAll() {
return users;
}
public User findByName(String name) {
if (name == null) throw new RuntimeException("請輸入正確的ID!");
return (User) users.stream().filter(t -> name.equals(t.name)).findFirst().orElse(null);
}
}
interface JsonFilter { // Target
String allToJson();
String findByNameToJson(String id);
}
class JsonFilterAdapter extends FilterImpl implements JsonFilter { // Adapter
public JsonFilterAdapter(List<User> users) {
super(users);
}
public String allToJson() {
List<User> users = super.findAll();
return new Gson().toJson(users);
}
public String findByNameToJson(String name) {
User user = super.findByName(name);
return new Gson().toJson(user);
}
}
代碼修改后,我們來看一個客戶端的使用案例,
class client {
public static void main(String[] args) {
List<User> users = new ArrayList<>();
users.add(new User("張三", "男", 19));
users.add(new User("李四", "男", 35));
users.add(new User("小紅", "女", 21));
JsonFilterAdapter jfa = new JsonFilterAdapter(users);
String allUser = jfa.allToJson();
String user = jfa.findByNameToJson("張三");
System.out.printf("%s%n%s", allUser, user);
}
}
使用了類配接器確實讓我們可以在不修改原有兩個模塊的情況下,以增加一個配接器類為代價整合兩大模塊,但,由于類配接器需要繼承結構中的Adaptee,且在客戶端中的使用我們也能看出雖然我們能夠呼叫繼承的方法但沒有直接使用,既然不使用為啥繼承呢?
這就引出了新的問題:我們是否有必要使用繼承來實作配接器,如答案是否,那不使用繼承我們又如何設計配接器呢,這就得看接下來的優化了,
修改版v2(物件配接器)(推薦)
我們的問題,物件配接器可以解決!!!
class User {
String name;
String sex;
int age;
// 剩下的屬性就不寫了,都是廢話沒啥意義
public User(String name, String sex, int age) {
this.name = name;
this.sex = sex;
this.age = age;
}
}
interface Filter {
List<User> findAll();
User findByName(String name);
}
class FilterImpl implements Filter { // Adaptee
List<User> users;
public FilterImpl(List<User> users) {
this.users = users;
}
public List<User> findAll() {
return users;
}
public User findByName(String name) {
if (name == null) throw new RuntimeException("請輸入正確的ID!");
return (User) users.stream().filter(t -> name.equals(t.name)).findFirst().orElse(null);
}
}
interface JsonFilter { // Target
String allToJson();
String findByNameToJson(String id);
}
class JsonFilterAdapter implements JsonFilter { // 不在繼承FilterImpl
private Filter filter; // 繼承 → 聚合
public JsonFilterAdapter(List<User> users) {
this.filter = new FilterImpl(users);
}
public String allToJson() {
List<User> users = filter.findAll();
return new Gson().toJson(users);
}
public String findByNameToJson(String name) {
User user = filter.findByName(name);
return new Gson().toJson(user);
}
}
我們再看看客戶端的呼叫代碼,
class client {
public static void main(String[] args) {
List<User> users = new ArrayList<>();
users.add(new User("張三", "男", 19));
users.add(new User("李四", "男", 35));
users.add(new User("小紅", "女", 21));
JsonFilterAdapter jfa = new JsonFilterAdapter(users);
String allUser = jfa.allToJson();
String user = jfa.findByNameToJson("張三");
System.out.printf("%s%n%s", allUser, user);
}
}
呼叫代碼完全沒有變化,但實際上已經無法在客戶端中呼叫Filter中定義的方法了,這使得Adapter類的都耦合度更低,有利于使用和維護,
總結
優點
- 使用配接器模式,不需要對于現有模塊修改,符合開閉原則,
- 可以針對現有的模塊創建多種多樣的配接器,而客戶端只需要呼叫配接器即可,讓客戶端與現有的多個模塊解耦,防止日后模塊修改時客戶端也需要隨之修改,
缺點
- 隨著配接器類的加入,現有的系統將越發復雜,
- 增加了開發人員對于系統的理解難度,
適用場景
- 需要整合兩個不兼容介面的場景,
本文來自博客園,作者:buzuweiqi,轉載請注明原文鏈接:https://www.cnblogs.com/buzuweiqi/p/16747565.html
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/510898.html
標籤:其他
上一篇:Java學習筆記
