中介者模式
定义
定义一个中介对象来封装一系列对象之间的交互,使原有对象之间的耦合松散,且可以独立地改变它们之间的交互。中介者模式又叫调停模式,它是迪米特法则的典型应用。
特点
优点
- 类之间职责清晰,符合迪米特法则;
- 对象之间的耦合性降低,使得对象易于独立地被复用;
- 将对象间的一对多关联转变为一对一的关联,提高系统的灵活性,使得系统易于维护和扩展。
缺点
中介这模式将原本多个对象直接的相互依赖变成了中介者和对个对象类的依赖关系。当同事类越多时,中介者就会越臃肿,变得复杂难以维护。
结构与实现
结构
主要角色:
- 抽象中介者角色:它是中介者的接口,提供了同事对象注册和转发同事对象信息的抽象方法;
- 具体中介者角色:实现中介者接口,定义一个List来管理同事对象,协调各个同事角色之间的交互关系,因此它依赖于同事角色;
- 抽象同事类角色:定义同事类的接口,保存中介者对象,提供同事对象交互的抽象方法,实现所有的相互影响的同事类的公共功能;
- 具体同事类角色:是抽象同事类的实现者,当需要与其他同事对象交互时,由中介者对象负责后续的交互。
类图:
实现
// 抽象中介者
abstract class Mediator {
public abstract void register(Colleague colleague);
public abstract void replay(Colleague colleague);// 转发
}
// 具体中介者
class ConcreteMediator extends Mediator {
private List<Colleague> colleagues = new ArrayList<>();
public void register(Colleague colleague) {
if (!colleagues.contains(colleague)) {
colleagues.add(colleague);
colleague.setMedium(this);
}
}
public void replay(Colleague cl) {
for (Colleague colleague: colleagues) {
if (!colleague.equals(cl)) {
colleague.reveive();
}
}
}
}
// 抽象同事类
abstract class Colleague {
protected Mediator mediator;
public void setMedium(Mediator mediator) {
this.mediator = mediator;
}
public abstract void receive();
public abstract void send();
}
// 具体实现类
class ConcreteColleagueA extends Colleague {
public void receive() {
// 具体同事类A实现逻辑
}
public void send() {
// 具体同事类A转发
media.replay();
}
}
class ConcreteColleagueB extends Colleague {
public void receive() {
// 具体同事类B实现逻辑
}
public void send() {
media.replay();
}
}
// 调用客户端
class MediaPattern {
public static void main(String[] args) {
Mediator md = new ConcreteMediator();
Colleague c1, c2;
c1 = new ConcreteColleagueA();
c2 = new ConcreteColleagueB();
md.register(c1);
md.register(c2);
c1.send();
c2.send();
}
}
应用场景
- 当对象之间存在复杂的网状结构关系而导致依赖关系混乱且难以复用时;
- 当想创建一个运行于多个类之间的对象,又不想生成新的子类时。
模式拓展
通常通过一下方法拓展:
- 不定义中介者接口,把具体中介者对象实现成为单例;
- 同事对象不持有中介者,而是在需要的时候直接获取中介者并调用。