设计模式之中介模式(三分钟学会一个设计模式)

中介模式(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/ )

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