中介者模式
1、模式动机
- 在用户与用户直接聊天的设计方案中,用户对象之间存在很强的关联性,将导致系统出现如下问题:
系统结构复杂:对象之间存在大量的相互关联和调用,若有一个对象发生变化,则需要跟踪和该对象关联的其他所有对象,并进行适当处理。
对象可重用性差:由于一个对象和其他对象具有很强的关联,若没有其他对象的支持,一个对象很难被另一个系统或模块重用,这些对象表现出来更像一个不可分割的整体,职责较为混乱。
系统扩展性低:增加一个新的对象需要在原有相关对象上增加引用,增加新的引用关系也需要调整原有对象,系统耦合度很高,对象操作很不灵活,扩展性差。 - 在面向对象的软件设计与开发过程中,根据“单一职责原则”,我们应该尽量将对象细化,使其只负责或呈现单一的职责。
对于一个模块,可能由很多对象构成,而且这些对象之间可能存在相互的引用,为了减少对象两两之间复杂的引用关系,使之成为一个松耦合的系统,我们需要使用中介者模式,这就是中介者模式的模式动机。
2、模式定义
中介者模式(Mediator Pattern)定义:用一个中介对象来封装一系列的对象交互,中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。中介者模式又称为调停者模式,它是一种对象行为型模式。
模式结构
- 中介者模式包含如下角色:
Mediator: 抽象中介者
ConcreteMediator: 具体中介者
Colleague: 抽象同事类
ConcreteColleague: 具体同事类
3、模式分析
中介者模式可以使对象之间的关系数量急剧减少:

- 中介者承担两方面的职责:
中转作用(结构性):通过中介者提供的中转作用,各个同事对象就不再需要显式引用其他同事,当需要和其他同事进行通信时,通过中介者即可。该中转作用属于中介者在结构上的支持。
协调作用(行为性):中介者可以更进一步的对同事之间的关系进行封装,同事可以一致地和中介者进行交互,而不需要指明中介者需要具体怎么做,中介者根据封装在自身内部的协调逻辑,对同事的请求进行进一步处理,将同事成员之间的关系行为进行分离和封装。该协调作用属于中介者在行为上的支持。
4、模式实例与分析
- 实例:虚拟聊天室
某论坛系统欲增加一个虚拟聊天室,允许论坛会员通过该聊天室进行信息交流,普通会员(CommonMember)可以给其他会员发送文本信息,钻石会员(DiamondMember)既可以给其他会员发送文本信息,还可以发送图片信息。该聊天室可以对不雅字符进行过滤,如“日”等字符;还可以对发送的图片大小进行控制。用中介者模式设计该虚拟聊天室。

package 中介者模式2;
//抽象聊天室
//抽象中介者
public abstract class AbstractChatroom {
public abstract void register(Member member);
public abstract void sendText(String from,String to,String message);
public abstract void sendImage(String from,String to,String Image);
}
package 中介者模式2;
import java.util.Hashtable;
//具体中介者类
public class ChatGroup extends AbstractChatroom {
private Hashtable members = new Hashtable();
@Override
public void register(Member member) {
if(!members.contains(member)){
members.put(member.getName(),member);
member.setChatroom(this);
}
}
@Override
public void sendText(String from, String to, String message) {
Member member = (Member) members.get(to);
String newMessage = message;
newMessage = message.replaceAll("日","*");
member.receiveText(from,newMessage);
}
@Override
public void sendImage(String from, String to, String image) {
Member member = (Member) members.get(to);
//模拟图片大小判断
if(image.length()>5){
System.out.println("图片太大,发送失败");
}else{
member.receiveImage(from,image);
}
}
}
package 中介者模式2;
//抽象会员类
public abstract class Member {
protected AbstractChatroom chatroom;
protected String name;
public Member(String name){
this.name =name;
}
public AbstractChatroom getChatroom() {
return chatroom;
}
public void setChatroom(AbstractChatroom chatroom) {
this.chatroom = chatroom;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public abstract void sendText(String to,String message);
public abstract void sendImage(String to,String image);
public void receiveText(String from,String message){
System.out.println(from +"发送文本稿给"+this.name + ",内容为:"+message);
}
public void receiveImage(String from,String image){
System.out.println(from+"发送照片给给"+this.name + ",内容为:" +image);
}
}
package 中介者模式2;
//普通会员类
public class CommonMember extends Member {
public CommonMember(String name) {
super(name);
}
@Override
public void sendText(String to, String message) {
System.out.println("普通会员发送消息:");
chatroom.sendText(name,to,message);
}
@Override
public void sendImage(String to, String image) {
System.out.println("普通会员不能发送照片!");
}
}
package 中介者模式2;
//钻石会员
public class DiamondMember extends Member {
public DiamondMember(String name) {
super(name);
}
@Override
public void sendText(String to, String message) {
System.out.println("钻石会员发送消息:");
chatroom.sendText(name,to,message);
}
@Override
public void sendImage(String to, String image) {
System.out.println("钻石会员发送照片:");
chatroom.sendImage(name,to,image);
}
}
package 中介者模式2;
public class 虚拟聊天室 {
public static void main(String[] args) {
AbstractChatroom happyChat = new ChatGroup();
Member member1,member2,member3,member4,member5;
member1 = new DiamondMember("张三");
member2 = new DiamondMember("李四");
member3 = new DiamondMember("王五");
member4 = new CommonMember("Dinny");
member5 = new CommonMember("Janny");
happyChat.register(member1);
happyChat.register(member2);
happyChat.register(member3);
happyChat.register(member4);
happyChat.register(member5);
member1.sendText("李四","李四,你好!");
member2.sendText("张三","张三,你好!");
member1.sendText("李四","今天天气不错,有日");
member2.sendImage("张三","一张很大很大的太阳");
member2.sendImage("张三","太阳");
member3.sendText("Dinny","有问题吗?");
member3.sendText("Janny","有问题吗?");
member4.sendText("王五","没有了,谢谢");;
member5.sendText("王五","我也没有了");
member5.sendImage("王五","谢谢");
}
}
- 结果展示:

5、模式的优缺点
- 中介者模式的优点
简化了对象之间的交互。
将各同事解耦。
减少子类生成。
可以简化各同事类的设计和实现。 - 中介者模式的缺点
在具体中介者类中包含了同事之间的交互细节,可能会导致具体中介者类非常复杂,使得系统难以维护。
模式使用环境
在以下情况下可以使用中介者模式:
- 系统中对象之间存在复杂的引用关系,产生的相互依赖关系结构混乱且难以理解。
- 一个对象由于引用了其他很多对象并且直接和这些对象通信,导致难以复用该对象。
- 想通过一个中间类来封装多个类中的行为,而又不想生成太多的子类。可以通过引入中介者类来实现,在中介者中定义对象交互的公共行为,如果需要改变行为则可以增加新的中介者类。
6、模式应用
(1) 中介者模式在事件驱动类软件中应用比较多,在设计GUI应用程序时,组件之间可能存在较为复杂的交互关系,一个组件的改变将影响与之相关的其他组件,此时可以使用中介者模式来对组件进行协调。
(2) MVC 是Java EE 的一个基本模式,此时控制器Controller 作为一种中介者,它负责控制视图对象View和模型对象Model之间的交互。如在Struts中,Action就可以作为JSP页面与业务对象之间的中介者。
- 中介者模式与迪米特法则
在中介者模式中,通过创造出一个中介者对象,将系统中有关的对象所引用的其他对象数目减少到最少,使得一个对象与其同事之间的相互作用被这个对象与中介者对象之间的相互作用所取代。因此,中介者模式就是迪米特法则的一个典型应用。 - 中介者模式与GUI开发
中介者模式可以方便地应用于图形界面(GUI)开发中,在比较复杂的界面中可能存在多个界面组件之间的交互关系。
对于这些复杂的交互关系,有时候我们可以引入一个中介者类,将这些交互的组件作为具体的同事类,将它们之间的引用和控制关系交由中介者负责,在一定程度上简化系统的交互,这也是中介者模式的常见应用之一。
7、模式总结
中介者模式用一个中介对象来封装一系列的对象交互,中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。中介者模式又称为调停者模式,它是一种对象行为型模式。
中介者模式包含四个角色:抽象中介者用于定义一个接口,该接口用于与各同事对象之间的通信;具体中介者是抽象中介者的子类,通过协调各个同事对象来实现协作行为,了解并维护它的各个同事对象的引用;抽象同事类定义各同事的公有方法;具体同事类是抽象同事类的子类,每一个同事对象都引用一个中介者对象;每一个同事对象在需要和其他同事对象通信时,先与中介者通信,通过中介者来间接完成与其他同事类的通信;在具体同事类中实现了在抽象同事类中定义的方法。
通过引入中介者对象,可以将系统的网状结构变成以中介者为中心的星形结构,中介者承担了中转作用和协调作用。中介者类是中介者模式的核心,它对整个系统进行控制和协调,简化了对象之间的交互,还可以对对象间的交互进行进一步的控制。
中介者模式的主要优点在于简化了对象之间的交互,将各同事解耦,还可以减少子类生成,对于复杂的对象之间的交互,通过引入中介者,可以简化各同事类的设计和实现;中介者模式主要缺点在于具体中介者类中包含了同事之间的交互细节,可能会导致具体中介者类非常复杂,使得系统难以维护。
中介者模式适用情况包括:系统中对象之间存在复杂的引用关系,产生的相互依赖关系结构混乱且难以理解;一个对象由于引用了其他很多对象并且直接和这些对象通信,导致难以复用该对象;想通过一个中间类来封装多个类中的行为,而又不想生成太多的子类。
本文详细介绍了软件设计模式中的中介者模式,探讨了其模式动机、定义、结构及优缺点。通过虚拟聊天室的实例说明了如何使用中介者模式简化对象间的交互,以及在GUI开发中的应用。中介者模式能降低系统耦合度,提高对象的可重用性,但可能导致中介者类过于复杂。
&spm=1001.2101.3001.5002&articleId=106020606&d=1&t=3&u=455fe8a1266e4010942db5a1e4881eaa)

被折叠的 条评论
为什么被折叠?



