中介模式(Mediator)又稱之為調停模式。mediator [ˈmiːdieɪtə(r)] n. 調停者;斡旋者;解決紛爭的人(或機構); 本意就是解決糾紛的中間人它是面向對象六大原則中最少知道原則的一個典型應用。(關於面向對象六大原則,可看前文:https://www.cnblogs.com/ ...
中介模式(Mediator)又稱之為調停模式。
mediator [ˈmiːdieɪtə(r)]
n. 調停者;斡旋者;解決紛爭的人(或機構); 本意就是解決糾紛的中間人
它是面向對象六大原則中最少知道原則的一個典型應用。
(關於面向對象六大原則,可看前文:https://www.cnblogs.com/jilodream/p/5353512.html)
大概意思就是類設計時與外界儘量減低耦合,儘量少的依賴其他類,這樣就會降低類後期修改的風險。
官方的的定義如下:用一個中介對象來封裝一系列的對象交互。中介對象使得其他各個對象不再需要顯示的相互引用。使整體的耦合更加鬆散,而且可以改變獨立的改變他們之間的交互。
它是面向對象的23種設計模式中的一種,屬於行為模式的範圍。
中介模式大概就是這樣,比如你去買房,不論是價格討論,還是房子質量,都通過中介來進行,儘管中介會包裝一些邏輯,但是買家不需要管理所有賣家的信息,保留賣家的聯繫方式。賣家呢也不需要管理買家的信息,保留買家的聯繫方式。儘管是買賣房子,但是買家和賣家之間都是通過中介來進行交互,並不直接溝通。這樣做的好處就是買賣雙方都減輕了自己的工作負擔,不需要牢記對方的相關特性,所有這些都交給中介來維護。
比如沒有加入中介對象時,我們A/B 雙方大概需要維護m*n依賴條,如果雙方還要互相依賴,則是2*m*n條依賴,當A、B雙方的實例越來越多時,情況也會越來越複雜。
當我們加入中介對象時,則只需要維護A/B和中介對象之間的依賴,約2(m+n)條依賴,(防盜連接:本文首發自http://www.cnblogs.com/jilodream/ )
區別大概是下麵這個樣子
直接耦合
通過中介者組織
有些人會說AB 對象我就互相調用一下,需要搞這麼複雜麽?
如果只是簡單調用,當然沒必要引入中介類,當然是怎麼簡單怎麼來,但是倘若業務未來(或者已經)依賴很複雜,就應該儘早引入中介類,降低類之間不必要的耦合。
來看一個例子,調用雙方是
銀行和企業,他們可以互相給對方發消息。我們通過中介模式來組織整體結構:
銀行介面:
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 /** 4 * @discription 5 */ 6 public interface Bank { 7 void sendMsg(String msg); 8 9 void receiveMsg(String msg); 10 11 void register(UnionPay unionPay); 12 }
工商銀行實現
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 import lombok.extern.slf4j.Slf4j; 4 5 /** 6 * @discription 7 */ 8 @Slf4j 9 public class ICBCBank implements Bank { 10 11 private UnionPay unionPay; 12 13 @Override 14 public void sendMsg(String msg) { 15 log.warn("工商銀行發送消息到企業:" + msg); 16 unionPay.sendCompany(msg); 17 } 18 19 @Override 20 public void receiveMsg(String msg) { 21 log.warn("工商銀行收到企業消息:" + msg); 22 } 23 24 @Override 25 public void register(UnionPay unionPay) { 26 this.unionPay = unionPay; 27 } 28 }
建設銀行實現
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 import lombok.extern.slf4j.Slf4j; 4 5 /** 6 * @discription 7 */ 8 @Slf4j 9 public class CCBBank implements Bank { 10 11 private UnionPay unionPay; 12 13 @Override 14 public void sendMsg(String msg) { 15 log.warn("建設銀行發送消息到企業:" + msg); 16 unionPay.sendCompany(msg); 17 } 18 19 @Override 20 public void receiveMsg(String msg) { 21 log.warn("建設銀行收到企業消息:" + msg); 22 } 23 24 @Override 25 public void register(UnionPay unionPay) { 26 this.unionPay = unionPay; 27 } 28 }
企業介面
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 public interface Company { 4 void sendMsg(String msg); 5 6 void receiveMsg(String msg); 7 8 void register(UnionPay unionPay); 9 }
千度公司
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 import lombok.extern.slf4j.Slf4j; 4 5 /** 6 * @discription 7 */ 8 @Slf4j 9 public class QianDuCompany implements Company { 10 private UnionPay unionPay; 11 12 @Override 13 public void sendMsg(String msg) { 14 log.warn("千度公司發送消息到銀行:" + msg); 15 unionPay.sendBank(msg); 16 } 17 18 @Override 19 public void receiveMsg(String msg) { 20 log.warn("千度公司收到銀行消息:" + msg); 21 } 22 23 @Override 24 public void register(UnionPay unionPay) { 25 this.unionPay = unionPay; 26 } 27 }
大米公司
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 import lombok.extern.slf4j.Slf4j; 4 5 /** 6 * @discription 7 */ 8 @Slf4j 9 public class DaMiCompany implements Company { 10 private UnionPay unionPay; 11 12 @Override 13 public void sendMsg(String msg) { 14 log.warn("大米公司發送消息到銀行:" + msg); 15 unionPay.sendBank(msg); 16 } 17 18 @Override 19 public void receiveMsg(String msg) { 20 log.warn("大米公司收到銀行消息:" + msg); 21 } 22 23 @Override 24 public void register(UnionPay unionPay) { 25 this.unionPay = unionPay; 26 } 27 }
中介類
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 import com.alibaba.nacos.shaded.com.google.common.collect.Lists; 4 5 import java.util.List; 6 7 /** 8 * @discription 9 */ 10 public class UnionPay { 11 private List<Bank> bankList = Lists.newArrayList(); 12 13 private List<Company> companyList = Lists.newArrayList(); 14 15 public void register(Object... components) { 16 for (Object component : components) { 17 if (component instanceof Company) { 18 Company company = (Company) component; 19 companyList.add(company); 20 company.register(this); 21 } 22 if (component instanceof Bank) { 23 Bank bank = (Bank) component; 24 bankList.add(bank); 25 bank.register(this); 26 } 27 } 28 } 29 30 public void sendBank(String msg) { 31 for (Bank bank : bankList) { 32 bank.receiveMsg(msg); 33 } 34 } 35 36 public void sendCompany(String msg) { 37 for (Company company : companyList) { 38 company.receiveMsg(msg); 39 } 40 } 41 }
主類
1 package com.example.demo.learn.pattern.behavior.mediator; 2 3 /** 4 * @discription 5 */ 6 public class PatternMain { 7 public static void main(String[] args) { 8 Bank ccbBank = new CCBBank(); 9 Bank icbcBank = new ICBCBank(); 10 Company qianDuCompany = new QianDuCompany(); 11 Company daMiCompany = new DaMiCompany(); 12 UnionPay unionPay = new UnionPay(); 13 unionPay.register(ccbBank, icbcBank, qianDuCompany, daMiCompany); 14 ccbBank.sendMsg("歡迎各位企業來建設銀行貸款!"); 15 icbcBank.sendMsg("歡迎各位企業來工行洽談合作!"); 16 daMiCompany.sendMsg("哪家銀行目前有低息企業貸款?"); 17 daMiCompany.sendMsg("哪家銀行目前有工資卡優惠活動?"); 18 } 19 }
運行主類後,效果如下:
18:58:03.965 [main] WARN com.example.demo.learn.pattern.behavior.mediator.CCBBank - 建設銀行發送消息到企業:歡迎各位企業來建設銀行貸款! 18:58:03.969 [main] WARN com.example.demo.learn.pattern.behavior.mediator.QianDuCompany - 千度公司收到銀行消息:歡迎各位企業來建設銀行貸款! 18:58:03.969 [main] WARN com.example.demo.learn.pattern.behavior.mediator.DaMiCompany - 大米公司收到銀行消息:歡迎各位企業來建設銀行貸款! 18:58:03.970 [main] WARN com.example.demo.learn.pattern.behavior.mediator.ICBCBank - 工商銀行發送消息到企業:歡迎各位企業來工行洽談合作! 18:58:03.970 [main] WARN com.example.demo.learn.pattern.behavior.mediator.QianDuCompany - 千度公司收到銀行消息:歡迎各位企業來工行洽談合作! 18:58:03.970 [main] WARN com.example.demo.learn.pattern.behavior.mediator.DaMiCompany - 大米公司收到銀行消息:歡迎各位企業來工行洽談合作! 18:58:03.970 [main] WARN com.example.demo.learn.pattern.behavior.mediator.DaMiCompany - 大米公司發送消息到銀行:哪家銀行目前有低息企業貸款? 18:58:03.971 [main] WARN com.example.demo.learn.pattern.behavior.mediator.CCBBank - 建設銀行收到企業消息:哪家銀行目前有低息企業貸款? 18:58:03.971 [main] WARN com.example.demo.learn.pattern.behavior.mediator.ICBCBank - 工商銀行收到企業消息:哪家銀行目前有低息企業貸款? 18:58:03.971 [main] WARN com.example.demo.learn.pattern.behavior.mediator.DaMiCompany - 大米公司發送消息到銀行:哪家銀行目前有工資卡優惠活動? 18:58:03.971 [main] WARN com.example.demo.learn.pattern.behavior.mediator.CCBBank - 建設銀行收到企業消息:哪家銀行目前有工資卡優惠活動? 18:58:03.971 [main] WARN com.example.demo.learn.pattern.behavior.mediator.ICBCBank - 工商銀行收到企業消息:哪家銀行目前有工資卡優惠活動?
代碼類圖如下:
我們可以從類圖發現,銀行和企業並沒有直接關聯,他們都是直接耦合中介類,所有的請求和響應都是和中介類進行交互(防盜連接:本文首發自http://www.cnblogs.com/jilodream/ )
中介模式的優點是解耦:我們可以直接將第三方提供類進行組織,而不需要修改他們的代碼。
缺點也很明顯,業務依賴邏輯全部抽離到了中介類中,中介類會過於臃腫。
如果你覺得寫的不錯,歡迎轉載和點贊。 轉載時請保留作者署名jilodream/王若伊_恩賜解脫(博客鏈接:http://www.cnblogs.com/jilodream/