github地址:
题目背景
联合国在世界上就是中介者的角色,各国之间的关系复杂,类似不同的对象和对象之间的关系,这就要求对象之间需要知道其他所有对象,尽管将一个系统分割成许多对象通常可以增加其可复用性,但是对象间相互连接的激增优惠降低其可复用性。大量的连接使得一个对象不可能在没有其他对象的支持下工作,系统表现为一个不可分割的整体,所以,对系统的行为进行任何较大的改动就十分困难了。
这里可以应用“迪米特法则”,如果两个类不必彼此直接通讯,那么这两个类就不应该发生直接的相互作用。如果其中一个类需要调用另一个类的某一种方法的话,可以通过第三者转发这个调用。也就是说,国与国之间的关系,完全可以通过联合国这个中介者来维持,而不必直接通信。
中介者模式
中介者模式,用一个中介对象来封装一系列的对象交互。中介者使各个对象不需要显示的相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。[DP]
中介者模式主要包括以下几个类:
- Colleague叫做抽象同事类,而ConcretColleague是具体同事类,每个具体同事只知道自己的行为,而不了解其他同事类的情况,但它们却都认识中介者对象;
- Mediator是抽象中介者,定义了同事对象到中介者对象的接口,ConcretMediator是具体中介者对象,实现抽象类的方法,它需要知道所有具体同事类,并从具体同事接收消息,向具体同事对象发出命令。
from abc import ABCMeta, abstractmethodclass Mediator(): """ 抽象中介者 """ __metaclass__ = ABCMeta @abstractmethod def send(self, message, colleague): """ 定义一个抽象的发送消息方法,得到同事对象和发送消息 """ pass class Colleague(): """ 抽象同事类 """ __metaclass__ = ABCMeta def __init__(self, mediator): """ 构造方法,得到中介者对象 """ self.mediator = mediator class ConcreteMediator(Mediator): """ 具体中介者 """ def __init__(self): """ 需要了解所有的具体同事对象 """ self.colleague1 = None self.colleague2 = None def send(self, message, colleague): """ 重写发送消息的方法,根据对象作出选择判断,通知具体同事对象 """ if colleague == self.colleague1: self.colleague2.notify(message) else: self.colleague1.notify(message) class ConcreteColleague1(Colleague): """ 具体同事类1 """ def send(self, message): self.mediator.send(message, self) def notify(self, message): print("同事1得到消息:",message) class ConcreteColleague2(Colleague): """ 具体同事类2 """ def send(self, message): self.mediator.send(message, self) def notify(self, message): print("同事2得到消息:",message) def main(): m = ConcreteMediator() """ 让两个具体同事类认识中介者对象 """ c1 = ConcreteColleague1(m) c2 = ConcreteColleague2(m) """ 让中介者认识各个具体同事类 """ m.colleague1 = c1 m.colleague2 = c2 """ 具体同事类对象发送消息都是通过中介者转发 """ c1.send("吃饭了吗?") c2.send("还没,你请客么?") main()
同事2得到消息: 吃饭了吗?同事1得到消息: 还没,你请客么?
点评
由于有了Mediator,使得ConcreteColleague1和ConcreteColleague2在发送消息和接收消息时其实是通过中介者来完成,这就减少了它们之间的耦合度。
题目
用程序模拟,美国和伊拉克之间的对话都是通过联合国安理会作为中介来完成。
from abc import ABCMeta, abstractmethodclass UnitedNations(): """ 联合国机构,抽象中介者 """ __metaclass__ = ABCMeta @abstractmethod def send(self, message, colleague): """ 定义一个抽象的发送消息方法,得到同事对象和发送消息 """ pass class Country(): """ 国家类,抽象同事类 """ __metaclass__ = ABCMeta def __init__(self, mediator): """ 构造方法,得到中介者对象 """ self.mediator = mediator class UnitedNationsSecurityCouncil(Mediator): """ 联合国安全理事会,具体中介者 """ def __init__(self): """ 需要了解所有的具体同事对象 """ self.colleague1 = None self.colleague2 = None def send(self, message, colleague): """ 重写发送消息的方法,根据对象作出选择判断,通知具体同事对象 """ if colleague == self.colleague1: self.colleague2.notify(message) else: self.colleague1.notify(message) class USA(Colleague): """ 美国,具体同事类1 """ def send(self, message): self.mediator.send(message, self) def notify(self, message): print("美国 得到消息:",message) class Iraq(Colleague): """ 伊拉克,具体同事类2 """ def send(self, message): self.mediator.send(message, self) def notify(self, message): print("伊拉克 得到消息:",message) def main(): m = UnitedNationsSecurityCouncil() """ 让两个具体同事类认识中介者对象 """ c1 = USA(m) c2 = Iraq(m) """ 让中介者认识各个具体同事类 """ m.colleague1 = c1 m.colleague2 = c2 """ 具体同事类对象发送消息都是通过中介者转发 """ c1.send("吃饭了吗?") c2.send("还没,你请客么?") main()
伊拉克 得到消息: 吃饭了吗?美国 得到消息: 还没,你请客么?
点评
ConcretMediator这个类必须要知道所有ConcreteCollegue,这就使得ConcreteMediator责任太多,如果它出现问题,则整个系统都会出现问题。
中介者模式很容易在系统中应用,也很容易在系统中误用。当系统出现“多对多”交互复杂的对象群时,不要急于使用中介者模式,而要先反思你的系统在设计上是否合理。
中介者模式的优点:
- Mediator的出现减少了各个Colleague的耦合,使得可以独立地改变和复用各个Colleague类和Mediator;
- 由于把对象如何协作进行了抽象,将中介作为一个独立的概念并将其封装在一个对象中,这样关注的对象就从对象各自本身的行为转移到它们之间的交互上来,也就是站在一个更宏观的角度去看待系统。
中介者模式的缺点:
- 由于ConcreteMediator控制了集中化,于是就把交互复杂性变为了中介者的复杂性,这就使得中介者会变得比任何一个ConcreteColleague都复杂。