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

C++ 中介者模式详解

循以下设计原则:

  1. 单一职责原则:将对象间的交互集中到中介者中

  2. 迪米特法则:减少对象间的直接通信

  3. 开闭原则:可以新增中介者而不修改现有组件

主要优点

  1. 降低耦合:减少对象间的直接依赖

  2. 集中控制:将交互逻辑集中到中介者

  3. 简化维护:交互关系更容易理解和修改

  4. 复用性:组件可以更容易地被复用

模式结构

主要组件

  1. Mediator(中介者接口)

    • 定义与各组件通信的接口

  2. ConcreteMediator(具体中介者)

    • 实现中介者接口

    • 协调各组件之间的交互

    • 知道所有组件及其用途

  3. Component(组件)

    • 每个组件都知道其中介者对象

    • 组件间不直接通信,而是通过中介者

完整代码示例

#include <iostream>
#include <string>
#include <memory>
#include <vector>
#include <algorithm>// 前置声明
class Component;// ==================== 中介者接口 ====================
class Mediator {
public:virtual void notify(Component* sender, const std::string& event) const = 0;virtual ~Mediator() = default;
};// ==================== 组件基类 ====================
class Component {
protected:Mediator* mediator_;public:explicit Component(Mediator* mediator = nullptr) : mediator_(mediator) {}void setMediator(Mediator* mediator) {mediator_ = mediator;}virtual ~Component() = default;
};// ==================== 具体组件 ====================
class Button : public Component {std::string name_;public:explicit Button(const std::string& name, Mediator* mediator = nullptr): Component(mediator), name_(name) {}void click() {std::cout << name_ << " 被点击" << std::endl;if (mediator_) {mediator_->notify(this, "click");}}
};class TextBox : public Component {std::string text_;public:explicit TextBox(Mediator* mediator = nullptr) : Component(mediator) {}void setText(const std::string& text) {text_ = text;std::cout << "文本框内容设置为: " << text << std::endl;if (mediator_) {mediator_->notify(this, "text_changed");}}std::string getText() const { return text_; }
};class CheckBox : public Component {bool checked_ = false;public:explicit CheckBox(Mediator* mediator = nullptr) : Component(mediator) {}void check() {checked_ = !checked_;std::cout << "复选框状态: " << (checked_ ? "选中" : "未选中") << std::endl;if (mediator_) {mediator_->notify(this, checked_ ? "checked" : "unchecked");}}bool isChecked() const { return checked_; }
};// ==================== 具体中介者 ====================
class AuthenticationDialog : public Mediator {Button* loginButton_;Button* cancelButton_;TextBox* usernameTextBox_;TextBox* passwordTextBox_;CheckBox* rememberMeCheckBox_;public:AuthenticationDialog(Button* loginButton,Button* cancelButton,TextBox* usernameTextBox,TextBox* passwordTextBox,CheckBox* rememberMeCheckBox): loginButton_(loginButton),cancelButton_(cancelButton),usernameTextBox_(usernameTextBox),passwordTextBox_(passwordTextBox),rememberMeCheckBox_(rememberMeCheckBox) {loginButton_->setMediator(this);cancelButton_->setMediator(this);usernameTextBox_->setMediator(this);passwordTextBox_->setMediator(this);rememberMeCheckBox_->setMediator(this);}void notify(Component* sender, const std::string& event) const override {if (sender == loginButton_ && event == "click") {std::cout << "登录按钮点击 - 尝试登录..." << std::endl;if (usernameTextBox_->getText().empty() || passwordTextBox_->getText().empty()) {std::cout << "错误: 用户名和密码不能为空" << std::endl;} else {std::cout << "使用用户名 '" << usernameTextBox_->getText() << "' 和密码 '" << passwordTextBox_->getText() << "' 登录" << std::endl;if (rememberMeCheckBox_->isChecked()) {std::cout << "记住登录状态" << std::endl;}}} else if (sender == cancelButton_ && event == "click") {std::cout << "取消按钮点击 - 重置表单" << std::endl;usernameTextBox_->setText("");passwordTextBox_->setText("");rememberMeCheckBox_->check();} else if (sender == rememberMeCheckBox_ && (event == "checked" || event == "unchecked")) {std::cout << "记住我选项改变: " << event << std::endl;} else if ((sender == usernameTextBox_ || sender == passwordTextBox_) && event == "text_changed") {std::cout << "文本改变 - 验证输入..." << std::endl;}}
};// ==================== 客户端代码 ====================
int main() {std::cout << "=== 中介者模式演示: 登录对话框 ===" << std::endl;// 创建组件Button loginButton("登录");Button cancelButton("取消");TextBox usernameTextBox;TextBox passwordTextBox;CheckBox rememberMeCheckBox;// 创建中介者并关联组件AuthenticationDialog dialog(&loginButton, &cancelButton, &usernameTextBox, &passwordTextBox, &rememberMeCheckBox);// 模拟用户交互std::cout << "\n用户输入用户名和密码..." << std::endl;usernameTextBox.setText("user123");passwordTextBox.setText("password");std::cout << "\n用户点击记住我..." << std::endl;rememberMeCheckBox.check();std::cout << "\n用户点击登录按钮..." << std::endl;loginButton.click();std::cout << "\n用户点击取消按钮..." << std::endl;cancelButton.click();return 0;
}

模式变体

1. 观察者中介者

class ObserverMediator : public Mediator {std::vector<Component*> components_;public:void addComponent(Component* component) {components_.push_back(component);component->setMediator(this);}void notify(Component* sender, const std::string& event) override {for (auto component : components_) {if (component != sender) {// 通知其他组件}}}
};

2. 单例中介者

class SingletonMediator : public Mediator {static SingletonMediator* instance_;SingletonMediator() = default;public:static SingletonMediator* getInstance() {if (!instance_) {instance_ = new SingletonMediator();}return instance_;}void notify(Component* sender, const std::string& event) override {// 实现通知逻辑}
};

实际应用场景

  1. GUI开发:如示例中的对话框组件交互

  2. 聊天系统:聊天室作为用户间通信的中介

  3. 航空管制:塔台协调飞机间的通信

  4. 工作流系统:协调多个处理步骤

  5. 游戏开发:游戏对象间的交互协调

相关文章:

  • SAM论文学习
  • Windows系统安装VirtualBox-7及其以上的版本修改默认安装路径后提示
  • python标准库--heapq - 堆队列算法(优先队列)在算法比赛的应用
  • 【AI News | 20250512】每日AI进展
  • 使用Daemonset部署日志收集守护进程
  • 探索边缘计算:赋能物联网的未来
  • WEBSTORM前端 —— 第3章:移动 Web —— 第1节:平面转换、渐变
  • 快消品商超业务单据解决方案重塑KA商超、电商业务与SAP ERP协同效率
  • 动态人脸识别教学实训沙盘功能介绍
  • 扩展:React 项目执行 yarn eject 后的 package.json 变化详解及参数解析
  • Linux进程10-有名管道概述、创建、读写操作、两个管道进程间通信、读写规律(只读、只写、读写区别)、设置阻塞/非阻塞
  • Spark处理过程-转换算子和行动算子
  • Lodash isEqual 方法源码实现分析
  • Spring Cloud Sleuth 链路追踪
  • Java面试高阶篇:Spring Boot+Quarkus+Redis高并发架构设计与性能优化实战
  • ZYNQ笔记(二十):Clocking Wizard 动态配置
  • 【开源工具】深度解析:基于PyQt6的Windows时间校时同步工具开发全攻略
  • bazel迁移cmake要点及具体迁移工程示例(apollo radar)
  • 技术视界 | 青龙机器人训练地形详解(四):复杂地形精讲之斜坡
  • 智表 ZCELL 插件快速入门指南(原创)
  • 5年建成强化城市核心功能新引擎,上海北外滩“风景文化都是顶流”
  • 国内首例侵入式脑机接口系统前瞻性临床试验:受试者已能用意念玩游戏
  • 伊朗外长称正与美国进行“善意”的会谈
  • 体验中国传统文化、采购非遗文创,波兰游客走进上海市群艺馆
  • 呼和浩特推进新一轮国企重组整合:杜绝一项目一公司、一业务一公司
  • 中俄就应对美加征所谓“对等关税”等问题进行深入交流