中介者模式(Mediator)
用一个中介对象来封装一系列的对象交互,中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。
腾讯QQ(简称“QQ”)是腾讯公司开发的一款基于Internet的即时通信(IM)软件。标志是一只戴着红色围巾的小企鹅。QQ注册用户由1999年的2人(马化腾和张志东)到现在已经发展到上亿用户,2014年4月11日21点11分在线人数突破两亿,如今已成为腾讯公司的代表之作,是中国目前使用最广泛的交流软件。
我们都知道用QQ能够点对点地跟别人聊天,也能在讨论组或者QQ群里面一人一句的聊天,如果点对点的聊天是每个人即作为服务器也作为客户端,发送消息和接收消息的话。那多人聊天采用这种结构就会相当的复杂,那我们该怎么办呢?加入一个中介者角色,充当通信服务器,每个人都作为客户端与服务器通信。下面我们就来看看具体实现:
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Collections; namespace Mediator { //用户接口 public interface IUser { //接收消息 void receiveMessage(String message); //发送消息 void sendMessage(String message); //获得发送的消息内容 string getMessage(); } //中介者接口 public interface IMediator { //注册用户信息 void regist(IUser user); //发送消息给所有人 void notifyAllMessage(IUser user); } public class ConcreteMediator : IMediator { private IList<IUser> list = new List<IUser>(); public void regist(IUser user) { if (user!=null && !list.Contains(user)){ list.Add(user); } } public void notifyAllMessage(IUser user) { for (int i = 0; i < list.Count; i++) { if (!list[i].Equals (user)) { list[i].receiveMessage(user.getMessage()); } } } } //抽象用户信息 public abstract class AbstractUser : IUser { protected IMediator mediator; protected String name; protected String message; public AbstractUser(IMediator mediator, string name) { this.mediator = mediator; this.name = name; } public void sendMessage(String message) { this.message = message; Console.WriteLine(this.name + " 说:" + this.message); mediator.notifyAllMessage(this); } public abstract void receiveMessage(string message); public string getMessage() { return message; } } //用户A public class UserA : AbstractUser { public UserA(IMediator mediator, string name) : base(mediator, name) { mediator.regist(this); } public override void receiveMessage(string message) { Console.WriteLine("User A Received Message :" + message); } } //用户B public class UserB : AbstractUser { public UserB(IMediator mediator, string name) : base(mediator, name) { mediator.regist(this); } public override void receiveMessage(string message) { Console.WriteLine("User B Received Message :" + message); } } //用户C public class UserC : AbstractUser { public UserC(IMediator mediator, string name) : base(mediator, name) { mediator.regist(this); } public override void receiveMessage(string message) { Console.WriteLine("User C Received Message :" + message); } } class Program { static void Main(string[] args) { //创建聊天室(中介者) IMediator mediator = new ConcreteMediator(); //创建用户信息 IUser userA = new UserA(mediator, "张三"); IUser userB = new UserB(mediator, "李四"); IUser userC = new UserC(mediator, "王五"); Console.WriteLine("------------------------------------------------------"); userA.sendMessage("大家好,我叫张三!初来乍到,请多多关照!"); Console.WriteLine(); Console.WriteLine("------------------------------------------------------"); userB.sendMessage("大家好,我叫李四!欢迎加入!"); Console.WriteLine(); Console.WriteLine("------------------------------------------------------"); userC.sendMessage("我是王五,欢迎欢迎!"); Console.WriteLine(); Console.WriteLine("------------------------------------------------------"); } } }
类图:
中介者模式包含如下几个角色:
Mediator(抽象中介者):它定义一个接口,该接口用于与各同事对象之间进行通信。
ConcreteMediator(具体中介者):具体中介者角色实现抽象中介者接口,具体协调各个同事类之间的协作。
Colleague(抽象同事类):定义与中介者之间沟通的接口。
ConcreteColleague(具体同事类):实现抽象同事角色接口,具体处理与中介者的通信行为。
主要优点:
1.中介者模式简化了对象之间的交互。一对多关系更容易理解、维护和扩展,将原本难以理解的网状结构转换成相对简单的星型结构。
2.中介者模式可将各同事对象解耦。
主要缺点:
在具体中介者类中包含了大量同事之间的交互细节,可能会导致具体中介者类非常复杂,使得系统难以维护。
适用场景:
1.一组对象以定义良好但是复杂的方式进行通信,产生的相互依赖关系结构混乱且难以理解。注意是多个对象之前相互依赖。
2.想定制一个分部在多个类中的行为,而不像生成太多的子类的场合。
相关模式:
1.外观:在外观中,外观角色是构成系统内部复杂子系统的单一窗口,对系统外部提供更高一级的接口。外观是单向的信息传达。
2.中介者:中介者角色充当着各个同事对象之间信息交互的中间角色。中介者是双向的信息通信。