中介模式(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/ )
中介模式的优点是解耦:我们可以直接将第三方提供类进行组织,而不需要修改他们的代码。
缺点也很明显,业务依赖逻辑全部抽离到了中介类中,中介类会过于臃肿。