中介者模式 (Mediator)
中介者模式 是一种行为型设计模式,它用一个中介对象来封装一组对象之间的交互。中介者通过协调多个对象之间的通信,避免对象之间的直接依赖,从而实现对象之间的松耦合。
意图
- 通过引入一个中介者对象,减少对象之间的直接依赖,从而降低耦合性。
- 中介者负责管理和协调对象之间的通信。
使用场景
- 对象之间的交互变得复杂 :
- 如果对象之间存在大量的直接引用和交互,可以使用中介者模式减少耦合。
- 希望解耦多个对象 :
- 通过中介者,避免对象之间的直接依赖。
- 需要集中控制交互 :
- 需要通过一个中心对象协调和控制所有对象的行为。
参与者角色
- 中介者接口 (Mediator)
- 定义对象之间通信的接口。
- 具体中介者 (ConcreteMediator)
- 实现中介者接口,负责具体的通信和协调。
- 同事类 (Colleague)
- 定义与中介者通信的接口,所有同事类通过中介者进行通信。
- 具体同事类 (ConcreteColleague)
- 实现同事类的行为,与中介者进行交互。
示例代码
以下代码展示了中介者模式的实现,模拟一个聊天室中的用户通过中介者进行通信的场景。
cpp
#include <iostream>
#include <string>
#include <vector>
#include <memory>
// 前置声明中介者
class Mediator;
// 同事类接口
class Colleague {
protected:
Mediator* mediator; // 中介者指针
std::string name; // 同事名称
public:
Colleague(Mediator* mediator, std::string name) : mediator(mediator), name(std::move(name)) {}
virtual ~Colleague() = default;
// 接收消息
virtual void receiveMessage(const std::string& sender, const std::string& message) = 0;
// 发送消息
virtual void sendMessage(const std::string& message) = 0;
std::string getName() const {
return name;
}
};
// 中介者接口
class Mediator {
public:
virtual ~Mediator() = default;
// 注册同事
virtual void addColleague(std::shared_ptr<Colleague> colleague) = 0;
// 转发消息
virtual void relayMessage(const std::string& sender, const std::string& message) = 0;
};
// 具体中介者
class ChatMediator : public Mediator {
private:
std::vector<std::shared_ptr<Colleague>> colleagues; // 同事列表
public:
void addColleague(std::shared_ptr<Colleague> colleague) override {
colleagues.push_back(std::move(colleague));
}
void relayMessage(const std::string& sender, const std::string& message) override {
for (const auto& colleague : colleagues) {
if (colleague->getName() != sender) {
colleague->receiveMessage(sender, message); // 转发消息给其他同事
}
}
}
};
// 具体同事类
class User : public Colleague {
public:
User(Mediator* mediator, std::string name) : Colleague(mediator, std::move(name)) {}
void receiveMessage(const std::string& sender, const std::string& message) override {
std::cout << name << " 收到来自 " << sender << " 的消息: " << message << "
";
}
void sendMessage(const std::string& message) override {
std::cout << name << " 发送消息: " << message << "
";
mediator->relayMessage(name, message); // 通过中介者发送消息
}
};
// 客户端代码
int main() {
// 创建中介者
auto chatMediator = std::make_shared<ChatMediator>();
// 创建用户
auto user1 = std::make_shared<User>(chatMediator.get(), "Alice");
auto user2 = std::make_shared<User>(chatMediator.get(), "Bob");
auto user3 = std::make_shared<User>(chatMediator.get(), "Charlie");
// 注册用户到中介者
chatMediator->addColleague(user1);
chatMediator->addColleague(user2);
chatMediator->addColleague(user3);
// 用户发送消息
user1->sendMessage("大家好!");
user2->sendMessage("你好,Alice!");
user3->sendMessage("欢迎加入聊天室!");
return 0;
}
代码解析
1. 中介者接口 (Mediator)
- 定义了对象之间通信的接口。
- 包含方法
addColleague
用于注册同事对象,relayMessage
用于转发消息。
cpp
class Mediator {
public:
virtual ~Mediator() = default;
virtual void addColleague(std::shared_ptr<Colleague> colleague) = 0;
virtual void relayMessage(const std::string& sender, const std::string& message) = 0;
};
2. 具体中介者 (ChatMediator)
- 实现中介者接口,负责管理同事对象并转发消息。
cpp
class ChatMediator : public Mediator {
private:
std::vector<std::shared_ptr<Colleague>> colleagues;
public:
void addColleague(std::shared_ptr<Colleague> colleague) override {
colleagues.push_back(std::move(colleague));
}
void relayMessage(const std::string& sender, const std::string& message) override {
for (const auto& colleague : colleagues) {
if (colleague->getName() != sender) {
colleague->receiveMessage(sender, message);
}
}
}
};
3. 同事类接口 (Colleague)
- 定义与中介者通信的接口。
- 包含方法
sendMessage
和receiveMessage
。
cpp
class Colleague {
protected:
Mediator* mediator;
std::string name;
public:
Colleague(Mediator* mediator, std::string name) : mediator(mediator), name(std::move(name)) {}
virtual ~Colleague() = default;
virtual void receiveMessage(const std::string& sender, const std::string& message) = 0;
virtual void sendMessage(const std::string& message) = 0;
std::string getName() const { return name; }
};
4. 具体同事类 (User)
- 实现同事类的行为,与中介者进行交互。
cpp
class User : public Colleague {
public:
User(Mediator* mediator, std::string name) : Colleague(mediator, std::move(name)) {}
void receiveMessage(const std::string& sender, const std::string& message) override {
std::cout << name << " 收到来自 " << sender << " 的消息: " << message << "
";
}
void sendMessage(const std::string& message) override {
std::cout << name << " 发送消息: " << message << "
";
mediator->relayMessage(name, message);
}
};
5. 客户端代码
- 客户端创建中介者和同事对象,并通过中介者实现对象之间的通信。
优缺点
优点
- 降低耦合 :
- 同事对象之间无需直接引用,通过中介者完成通信。
- 集中控制 :
- 通过中介者,可以集中管理和控制对象的交互。
- 灵活扩展 :
- 可以通过扩展中介者的功能,改变对象间的交互逻辑。
缺点
- 中介者复杂性增加 :
- 随着对象交互的增加,中介者可能变得复杂且难以维护。
- 单点故障 :
- 中介者是交互的核心,其故障会影响整个系统。
适用场景
- 对象间的交互复杂 :
- 希望通过引入中介者降低对象间的直接依赖。
- 需要集中控制交互逻辑 :
- 希望通过中介者管理对象之间的通信。
- 希望解耦多个对象 :
- 通过中介者实现对象之间的松耦合。
总结
中介者模式通过引入中介者对象,降低了对象之间的耦合性,同时集中管理了交互逻辑。它特别适用于对象间交互复杂且需要统一管理的场景。