当前位置: 首页 > news >正文

C++设计模式——Mediator中介者模式

一,中介者模式的定义

中介者模式是一种行为型设计模式。它通过一个中介者对象将多个对象之间的交互关系进行封装,使得对象之间的交互需要通过中介者对象来完成。该设计模式的结构很容易理解,以中介者为中心。

中介者模式的设计思想侧重于在对象之间增加一个用来调度的中介。

有了中介者模式,各个对象可以专注于各自的业务处理逻辑,而不需要关心通信的具体实现细节。

中介者模式在现实生活中的抽象实例:

航空管制系统:航空管制系统作为中介者,协调飞机、航空公司和机场的通信和协作。

交易系统:在金融领域,交易系统将银行、金融机构、客户等各个参与者进行协调,确保资金的安全快速转移。

买房中介:买房中介充当着买卖双方之间的桥梁和调解者的角色,确保双方利益的平衡和交易的顺利进行。

二,中介者模式的结构

中介者模式主要包含以下组件:

1.抽象中介者(Mediator):定义了对象之间相互通信的规则,定义了管理对象和消息通信的统一接口。

2.抽象同事对象(Colleague):是参与通信的各个对象,内部包含对中介者对象的引用。负责将消息发送给中介者,以及接收并处理中介者发来的消息。

3.具体中介者(Concrete Mediator):包含对抽象中介者的具体实现,负责协调各个对象之间的通信,协调的方式以转发消息为主。

4.具体同事对象(Concrete Colleague):包含对抽象同事对象的具体实现。它们之间通过调用中介者的接口进行通信,并接收和处理中介者转发给它们的消息。

组件之间的工作步骤如下:

1.初始化中介者对象。

2.各个同事对象与中介者关联,将中介者对象传递给各个同事对象。

3.同事对象与中介者通信,同事对象调用中介者对象提供的通信接口,由中介者负责将信息转发给目标同事对象。

对应UML类图:

三,中介者模式代码样例

#include <iostream>
#include <string>
#include <vector>class Colleague;class Mediator{
public:virtual void sendMessage(const std::string& msg, Colleague* colleague) = 0;virtual void addColleague(Colleague* colleague) = 0;
};class Colleague{
public:Colleague(Mediator* mediator) : mediator_(mediator) {}virtual void sendMessage(const std::string& message) = 0;virtual void receiveMessage(const std::string& message) = 0;
protected:Mediator* mediator_;
};class ConcreteMediator : public Mediator{
public:void sendMessage(const std::string& msg, Colleague* colleague) override{for (auto col : colleagues_) {if (col != colleague) {col->receiveMessage(msg);}}}void addColleague(Colleague* colleague) override {colleagues_.push_back(colleague);}
private:std::vector<Colleague*> colleagues_;
};class ConcreteColleague : public Colleague{
public:ConcreteColleague(Mediator* mediator) : Colleague(mediator) {}void sendMessage(const std::string& message) override {mediator_->sendMessage(message, this);}void receiveMessage(const std::string& message) override {std::cout << "Received message: " << message << std::endl;}
};int main() {Mediator* mediator = new ConcreteMediator();Colleague* colleague1 = new ConcreteColleague(mediator);Colleague* colleague2 = new ConcreteColleague(mediator);mediator->addColleague(colleague1);mediator->addColleague(colleague2);colleague1->sendMessage("Hello from colleague1");colleague2->sendMessage("Hello from colleague2");delete colleague1;delete colleague2;delete mediator;return 0;
}

运行结果:

Received message: Hello from colleague1
Received message: Hello from colleague2

四,中介者模式的应用场景

事件驱动架构:应用程序中,按钮点击等事件不需要直接关联所有处理响应的逻辑,而是通过一个“事件总线”或“消息中间件”来分发消息。

GUI用户界面:在UI组件间传递事件或更新状态时,可以使用中介者模式避免硬编码依赖。

分布式系统:分布式应用中设定一个集中式的服务器作为中介,协调客户端之间的交互。

消息队列:在异步通信场景,发送者和接收者通过一个消息中间件来传递信息,方便解耦和事务管理。

五,中介者模式的优缺点

中介者模式的优点:

降低了对象之间的耦合,易于维护。

可以实现对通信的集中控制。

方便随时修改和消息对应的事件处理。

在不改变原有对象的基础上,可以灵活添加新的消息类型。

中介者模式的缺点:

容易导致对系统的过度设计。

当对象很多时,中介者会变得复杂和难以管理。

通信期间需要额外的调度,性能开销大。

六,代码实战

Demo1:基于中介者模式实现的消息群发功能

#include <iostream>
#include <string>
#include <vector>class User;class Mediator {
public:virtual void sendMessage(const std::string& message, User* user) = 0;virtual void addUser(User* user) = 0;
};class User {
public:User(const std::string& name, Mediator* mediator){this->name = name;this->mediator = mediator;}const std::string& getName() const {return name;}void sendMessage(const std::string& message) {mediator->sendMessage(message, this);}virtual void receiveMsg(const std::string& message) = 0;
private:std::string name;Mediator* mediator;
};class ChatRoom : public Mediator {
public:void addUser(User* user) {users.push_back(user);}void sendMessage(const std::string& message, User* sender) override {for (User* user : users) {if (user != sender) {user->receiveMsg(message);}}}
private:std::vector<User*> users;
};class ChatUser : public User {
public:ChatUser(const std::string& name, Mediator* mediator) : User(name, mediator) {}void receiveMsg(const std::string& msg) override {std::cout << getName() << " received a message: " << msg << std::endl;}
};int main() {Mediator* chatRoom = new ChatRoom();User* user1 = new ChatUser("User1", chatRoom);User* user2 = new ChatUser("User2", chatRoom);User* user3 = new ChatUser("User3", chatRoom);chatRoom->addUser(user1);chatRoom->addUser(user2);chatRoom->addUser(user3);user1->sendMessage("Hello, everyone!");delete user1;delete user2;delete user3;delete chatRoom;return 0;
}

运行结果:

User2 received a message: Hello, everyone!
User3 received a message: Hello, everyone!

Demo2:模拟的聊天室

#include <iostream>
#include <string>
#include <vector>using namespace std;struct ChatRoom {virtual void broadcast(string from, string msg) = 0;virtual void message(string from, string to, string msg) = 0;
};struct Person {string m_name;ChatRoom* m_room{ nullptr };vector<string> m_chat_log;Person(string n) : m_name(n) {}void say(string msg) const {m_room->broadcast(m_name, msg);}void pm(string to, string msg) const {m_room->message(m_name, to, msg);}void receive(string from, string msg) {string s{ from + ": \"" + msg + "\"" };cout << "[" << m_name << "'s chat session]" << s << "\n";m_chat_log.emplace_back(s);}
};struct GoogleChat: ChatRoom
{vector<Person*> m_people;void broadcast(string from, string msg) {for (auto p : m_people)if (p->m_name != from)p->receive(from, msg);}void join(Person* p) {string join_msg = p->m_name + " joins the chat";broadcast("room", join_msg);p->m_room = this;m_people.push_back(p);}void message(string from, string to, string msg) {auto target = find_if(begin(m_people), end(m_people),[&](const Person* p) {return p->m_name == to;});if (target != end(m_people)) (*target)->receive(from, msg);}
};int main() {GoogleChat room;Person john{ "John" };Person jane{ "Jane" };room.join(&john);room.join(&jane);john.say("hi room");jane.say("oh, hey john");Person simon{ "Simon" };room.join(&simon);simon.say("hi everyone!");jane.pm("Simon", "glad you found us, simon!");return EXIT_SUCCESS;
}

运行结果:

[John's chat session]room: "Jane joins the chat"
[Jane's chat session]John: "hi room"
[John's chat session]Jane: "oh, hey john"
[John's chat session]room: "Simon joins the chat"
[Jane's chat session]room: "Simon joins the chat"
[John's chat session]Simon: "hi everyone!"
[Jane's chat session]Simon: "hi everyone!"
[Simon's chat session]Jane: "glad you found us, simon!"

七,参考阅读

https://www.geeksforgeeks.org/mediator-design-pattern/

https://www.patterns.dev/vanilla/mediator-pattern/

https://vishalchovatiya.com/posts/mediator-design-pattern-in-modern-cpp/

https://softwarepatterns.com/cpp/mediator-software-pattern-cpp-example

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • RAII 与 std::lock_guard 在 C++ 中的应用:自动化互斥锁管理与线程安全
  • 【JavaScript】数据结构之字典 哈希表
  • HTML5下<hr>标签的CSS样式定制
  • 关于Linux Makefile的一些函数知识foreach、if、call、origin、shell、error和warning
  • 如何使用myabtis log plugin插件展示出数据库查询语句
  • 【C++11】智能指针
  • STM32常用数据采集滤波算法
  • Java | Leetcode Java题解之第400题第N位数字
  • 决策树基础概论
  • Android 13 固定systemUI的状态栏为黑底白字,不能被系统应用或者三方应用修改
  • flutter widget.onPressed回调无效
  • 微软面向所有用户推出 Xbox Game Pass Standard
  • 四、链表————相关概念详解
  • 【数据结构与算法 | 灵神题单 | 合并链表篇】力扣2, 21, 445, 2816
  • Spring 循环依赖原理及解决方案
  • 4月23日世界读书日 网络营销论坛推荐《正在爆发的营销革命》
  • Angularjs之国际化
  • Bootstrap JS插件Alert源码分析
  • express如何解决request entity too large问题
  • k个最大的数及变种小结
  • Laravel 中的一个后期静态绑定
  • Linux各目录及每个目录的详细介绍
  • maven工程打包jar以及java jar命令的classpath使用
  • MySQL数据库运维之数据恢复
  • mysql中InnoDB引擎中页的概念
  • Python 基础起步 (十) 什么叫函数?
  • react-native 安卓真机环境搭建
  • Selenium实战教程系列(二)---元素定位
  • Spark VS Hadoop:两大大数据分析系统深度解读
  • 仿天猫超市收藏抛物线动画工具库
  • 紧急通知:《观止-微软》请在经管柜购买!
  • 目录与文件属性:编写ls
  • 微信小程序实战练习(仿五洲到家微信版)
  • 一文看透浏览器架构
  • 云大使推广中的常见热门问题
  • 我们雇佣了一只大猴子...
  • ​​​​​​​​​​​​​​Γ函数
  • # Redis 入门到精通(九)-- 主从复制(1)
  • # 日期待t_最值得等的SUV奥迪Q9:空间比MPV还大,或搭4.0T,香
  • ## 临床数据 两两比较 加显著性boxplot加显著性
  • #预处理和函数的对比以及条件编译
  • (poj1.3.2)1791(构造法模拟)
  • (rabbitmq的高级特性)消息可靠性
  • (Redis使用系列) Springboot 整合Redisson 实现分布式锁 七
  • (草履虫都可以看懂的)PyQt子窗口向主窗口传递参数,主窗口接收子窗口信号、参数。
  • (非本人原创)史记·柴静列传(r4笔记第65天)
  • (附源码)spring boot火车票售卖系统 毕业设计 211004
  • (十八)用JAVA编写MP3解码器——迷你播放器
  • (十三)Flink SQL
  • (一) storm的集群安装与配置
  • (一一四)第九章编程练习
  • (转)nsfocus-绿盟科技笔试题目
  • ... fatal error LINK1120:1个无法解析的外部命令 的解决办法
  • .md即markdown文件的基本常用编写语法
  • .net core IResultFilter 的 OnResultExecuted和OnResultExecuting的区别