设计模式专栏
1.定义:
中介者模式(Mediator Pattern)是用来降低多个对象和类之间的通信复杂性。这种模式提供了一个中介类,该类通常处理不同类之间的通信,并支持松耦合,使代码易于维护。中介者模式属于行为型模式。
2.使用场景
中介者模式很容易实现呢,但是也容易误用,不要着急使用,先要思考你的设计是否合理。
当对象之间的交互变多时,为了防止一个类会涉及修改其他类的行为,可以使用中介者模式,将系统从网状结构变为以中介者为中心的星型结构。
3.UML图示
Mediator :抽象中介者角色,定义了同事对象到中介者对象的接口。
ConcreteMediator:具体中介者角色,它从具体的同事对象接收消息,向具体同事发出命令。
Colleague:抽象同事角色,定义了中介者对象接口,它只知道中介者而不知道其他同事对象。
ConcreteColleague:具体同事角色,每个具体同事类都知道自己在小范围内的行为,而不知道它在大范围内的目的。
4.中介者模式的简单实现
中介者模式可以拿武侠来举例,江湖中门派众多,门派之前因为想法不同会有很多的利益冲突,这样就会带来无休止的纷争。为了江湖的安宁,大家推举出了一个大家都认可的武林盟主来对江湖纷争进行调停。
前段时间武当派和峨眉派的的弟子被大力金刚指所杀,大力金刚指是少林派的绝学,因为事情重大,而且少林派实力强大,武当派和峨眉派不能够直接去少林派去问罪,只能上报武林盟主由武林盟主出面进行调停,如下图所示。
- 抽象中介者角色
首先我们创建抽象中介者类,在这个例子中,它是一个武林联盟类,如下所示。
public abstract class WulinAlliance {
abstract void notice(String message,United united );
}
notice方法用于向门派发送通知,其中United为抽象同事类也就是门派类,接下来我们来创建它。
- 抽象同事角色
public class United {
protected WulinAlliance wulinAlliance;
public United(WulinAlliance wulinAlliance) {
this.wulinAlliance = wulinAlliance;
}
}
门派类(抽象同事类)会在构造方法中得到武林联盟类(抽象中介者类)。
- 具体同事角色
具体同事类包括武当派、峨眉派和少林派,如下所示。
public class Emei extends United {
public Emei(WulinAlliance wulinAlliance) {
super(wulinAlliance);
}
public void sendAlliance(String message) {
wulinAlliance.notice(message, this);
}
public void getNotice(String message) {
System.out.println("峨眉收到消息:" + message);
}
}
public class ShaoLin extends United {
public ShaoLin(WulinAlliance wulinAlliance) {
super(wulinAlliance);
}
public void sendAlliance(String message) {
wulinAlliance.notice(message, this);
}
public void getNotice(String message) {
System.out.println("少林收到消息:" + message);
}
}
public class Wudang extends United {
public Wudang(WulinAlliance wulinAlliance) {
super(wulinAlliance);
}
public void sendAlliance(String message) {
wulinAlliance.notice(message, this);
}
public void getNotice(String message) {
System.out.println("武当收到消息:" + message);
}
}
武当、峨眉和少林类都有两个方法,其中getNotice方法是自有方法,对于其他的门派(同事类)和武林联盟(中介者)没有依赖,只是用来接收武林盟主的通知。sendAlliance方法则是依赖方法,它必须通过武林盟主才能完成行为。
- 具体中介者角色
public class Champions extends WulinAlliance {
private Wudang wudang;
private ShaoLin shaolin;
private Emei emei;
public void setWudang(Wudang wudang) {
this.wudang = wudang;
}
public void setEmei(Emei emei) {
this.emei = emei;
}
public void setShaolin(ShaoLin shaolin) {
this.shaolin = shaolin;
}
@Override
public void notice(String message, United united) {
if (united == wudang) {
shaolin.getNotice(message);
} else if (united == emei) {
shaolin.getNotice(message);
} else if (united == shaolin) {
wudang.getNotice(message);
emei.getNotice(message);
}
}
}
武林盟主需要了解所有的门派,所以需要用setter来持有武当、峨眉和少林的引用。notice方法会根据不同门派发来的消息,转而通知给其他的门派。比如武当发来的消息,武林盟主就会将消息通知给少林。
- 客户端调用
public class Clinet {
public static void main(String[] args) {
Champions champions=new Champions();
Wudang wudang=new Wudang(champions);
ShaoLin shaolin=new ShaoLin(champions);
Emei emei=new Emei(champions);
champions.setWudang(wudang);
champions.setShaolin(shaolin);
champions.setEmei(emei);
wudang.sendAlliance("武当弟子被少林大力金刚指所杀");
emei.sendAlliance("峨眉弟子被少林大力金刚指所杀");
shaolin.sendAlliance("少林弟子绝不会做出这种事情");
}
}
首先创建武林盟主类Champions 并传入到各个门派类,接着调用武林盟主类的setter方法传入各个门派类,最后调用各个门派的sendAlliance方法通过武林盟主类向其他门派发送消息。
-
得到如下结果:
参考文献:
刘望舒的博客
Android源码设计模式