文章目录
- 概述
- 结构
- 优缺点及适用场景
- 案例实现
概述
中介者模式又叫调停模式,是一种行为模式,它定义一个中介角色来封装一系列对象之间的交互,使原有对象之间的耦合松散,且可以独立地改变它们之间的交互。
中介者模式是迪米特原则的经典体现:它要求一个对象应该对其他对象保持最少的了解。如果两个类不必彼此直接通信,那么这两个类就不应当发生直接的相互作用。如果其中的一个类需要调用另一个类的某一个方法的话,可以通过第三者转发这个调用。
中介者模式核心在于中介者这个角色,理解上可以参考现实中的中介者,中介就跟中间人一样,跟桥梁一样。以租房为例,试想一下,先不谈咱们能不能找到这么多房源,就算找到了能不能有那个精力一家一家谈对比。如果一个人冒冒失失的盲目的找,不仅花费的时间跟精力非常大,即使找到了也不是理想中的。这时候中介就发挥作用了。它会根据你的要求,帮你筛选出满足条件的房源,任你挑选,完了以后再跟房主进行联系。
代码的世界里,如果各个类之间互相依赖,每个类内部引入其他类,当类增多时就会形成网状依赖关系(下图左),这时代码的耦合度就很高;中介者模式可以很好的解耦这种依赖关系,所有类都只和中介者有关联,形成一种星状结构(下图右),通过中介者建立联系,而彼此不需要知道对方是谁,彼此独立,耦合性很低。
结构
中介者模式包含以下主要角色:
-
抽象中介者(
Mediator
)角色: 它是中介者的接口,提供了同事对象注册与转发同事对象信息的抽象方法; -
具体中介者(
ConcreteMediator
)角色: 实现中介者接口,定义一个 List 来管理同事对象,协调各个同事角色之间的交互关系,因此它依赖于同事角色; -
抽象同事类(
Colleague
)角色: 定义同事类的接口,保存中介者对象,提供同事对象交互的抽象方法,实现所有相互影响的同事类的公共功能; -
具体同事类(
Concrete Colleague
)角色: 是抽象同事类的实现者,当需要与其他同事对象交互时,由中介者对象负责后续的交互。
优缺点及适用场景
1、优点:
-
松散耦合
中介者模式通过把多个同事对象之间的交互封装到中介者对象里面,从而使得同事对象之间松散耦合,基本上可以做到互补依赖。这样一来,同事对象就可以独立地变化和复用,而不再像以前那样“牵一处而动全身”了。 -
集中控制交互
多个同事对象的交互,被封装在中介者对象里面集中管理,使得这些交互行为发生变化的时候,只需要修改中介者对象就可以了,当然如果是已经做好的系统,那么就扩展中介者对象,而各个同事类不需要做修改。 -
一对多关联转变为一对一的关联
没有使用中介者模式的时候,同事对象之间的关系通常是一对多的,引入中介者对象以后,中介者对象和同事对象的关系通常变成双向的一对一,这会让对象的关系更容易理解和实现。
2、缺点:
- 当同事类太多时,中介者的职责将很大,它会变得复杂而庞大,以至于系统难以维护。
3、适用场景
- 当系统中多个类相互耦合,形成网状结构时,可以使用中介者模式。
案例实现
【租房案例】 现在租房基本都是通过房屋中介,房主将房屋托管给房屋中介,而租房者从房屋中介获取房屋信息。房屋中介充当租房者与房屋所有者之间的中介者。
UML类图:
【抽象中介者角色】:
java">public abstract class Mediator {
//申明一个联络方法
public abstract void contact(String message, Person person);
}
中介机构【具体中介者角色】:
java">public class MediatorStructure extends Mediator {
//首先中介结构必须知道所有房主和租房者的信息
private HouseOwner houseOwner;
private Tenant tenant;
public HouseOwner getHouseOwner() {
return houseOwner;
}
public void setHouseOwner(HouseOwner houseOwner) {
this.houseOwner = houseOwner;
}
public Tenant getTenant() {
return tenant;
}
public void setTenant(Tenant tenant) {
this.tenant = tenant;
}
@Override
public void contact(String message, Person person) {
if (person == houseOwner) { //如果是房主,则租房者获得信息
tenant.getMessage(message);
} else { //反则是房主获得信息
houseOwner.getMessage(message);
}
}
}
【抽象同事类角色】:
java">public abstract class Person {
protected String name;
protected Mediator mediator;
public Person(String name, Mediator mediator){
this.name = name;
this.mediator = mediator;
}
}
房主、租客【具体同事类】:
java">public class HouseOwner extends Person {
public HouseOwner(String name, Mediator mediator) {
super(name, mediator);
}
//与中介者联系
public void contact(String message){
mediator.contact(message, this);
}
//获取信息
public void getMessage(String message){
System.out.println("房主" + name +"获取到的信息:" + message);
}
}
java">public class Tenant extends Person {
public Tenant(String name, Mediator mediator) {
super(name, mediator);
}
//与中介者联系
public void contact(String message){
mediator.contact(message, this);
}
//获取信息
public void getMessage(String message){
System.out.println("租房者" + name +"获取到的信息:" + message);
}
}
测试:
java">public class Client{
public static void main(String[] args) {
//一个房主、一个租房者、一个中介机构
MediatorStructure mediator = new MediatorStructure();
//房主和租房者只需要知道中介机构即可
HouseOwner houseOwner = new HouseOwner("张三", mediator);
Tenant tenant = new Tenant("李四", mediator);
//中介结构要知道房主和租房者
mediator.setHouseOwner(houseOwner);
mediator.setTenant(tenant);
tenant.contact("需要租三室的房子");
houseOwner.contact("我这有三室的房子,你需要租吗?");
}
}