新闻中心

c++怎么实现一个观察者模式_c++事件通知与解耦机制实现

2025-11-19
浏览次数:
返回列表
观察者模式通过Subject与Observer接口实现一对多依赖更新。1. Subject维护观察者列表,状态变化时通知所有Observer;2. Observer定义update纯虚函数接收通知;3. 使用智能指针与弱引用避免内存泄漏和悬挂指针;4. 加入互斥锁保障多线程安全;5. 适用于事件系统、日志广播等解耦场景。

c++怎么实现一个观察者模式_c++事件通知与解耦机制实现

观察者模式是一种行为设计模式,用于在对象之间建立一对多的依赖关系,当一个对象状态改变时,所有依赖它的对象都会收到通知并自动更新。在C++中,这种机制常被用来实现事件通知系统,达到模块间的解耦。

核心思路与角色划分

观察者模式包含两个主要角色:

  • Subject(被观察者):维护观察者列表,提供注册、注销和通知接口。
  • Observer(观察者):定义接收通知的接口,通常是一个纯虚函数。

通过抽象接口,Subject不需要知道具体是谁在监听,实现了发送方与接收方的解耦。

基础实现示例

下面是一个简洁的C++实现:

#include <iostream>
#include <vector>
#include <algorithm>
<p>// 观察者接口
class Observer {
public:
virtual ~Observer() = default;
virtual void update(const std::string& message) = 0;
};</p><p>// 被观察者
class Subject {
private:
std::vector<Observer*> observers;</p><p>public:
void attach(Observer* obs) {
observers.push_back(obs);
}</p><pre class='brush:php;toolbar:false;'>void detach(Observer* obs) {
    observers.erase(
        std::remove(observers.begin(), observers.end(), obs),
        observers.end()
    );
}

void notify(const std::string& message) {
    for (auto* obs : observers) {
        obs->update(message);
    }
}

};

定义两个具体的观察者:

class ConcreteObserverA : public Observer {
public:
    void update(const std::string& message) override {
        std::cout << "Observer A received: " << message << "\n";
    }
};
<p>class ConcreteObserverB : public Observer {
public:
void update(const std::string& message) override {
std::cout << "Observer B received: " << message << "\n";
}
};</p>

使用方式:

int main() {
    Subject subject;
    ConcreteObserverA observerA;
    ConcreteObserverB observerB;
<pre class='brush:php;toolbar:false;'>subject.attach(&observerA);
subject.attach(&observerB);

subject.notify("Hello Observers!");

subject.detach(&observerA);
subject.notify("Only B should see this.");

return 0;

}

小云雀 小云雀

剪映出品的AI视频和图片创作助手

小云雀 1949 查看详情 小云雀

改进:支持多线程与智能指针

上面的实现存在裸指针管理问题。实际项目中建议使用std::weak_ptr避免悬挂指针,并考虑线程安全。

改用std::shared_ptrstd::weak_ptr

#include <memory>
#include <list>
<p>class Observer;</p><p>class Subject {
private:
std::list<std::weak_ptr<Observer>> observers;
mutable std::mutex mtx; // 线程安全</p><p>public:
void attach(std::shared_ptr<Observer> obs) {
std::lock_guard<std::mutex> lock(mtx);
observers.remove_if([](const auto& w) { return w.expired(); });
observers.push_back(obs);
}</p><pre class='brush:php;toolbar:false;'>void notify(const std::string& message) {
    std::lock_guard<std::mutex> lock(mtx);
    observers.remove_if([&](const auto& w) {
        if (auto obs = w.lock()) {
            obs->update(message);
            return false;
        }
        return true; // 已过期,移除
    });
}

};

观察者也改为继承 enable_shared_from_this:

class Observer : public std::enable_shared_from_this<Observer> {
public:
    virtual ~Observer() = default;
    virtual void update(const std::string& message) = 0;
};

应用场景与优势

观察者模式适合以下场景:

  • GUI事件系统(按钮点击、窗口关闭)
  • 消息广播机制
  • 日志系统多输出目标
  • 游戏中的事件驱动逻辑

优点包括:

  • 松耦合:Subject不依赖具体Observer
  • 可扩展:新增观察者无需修改现有代码
  • 动态订阅:运行时灵活添加或移除监听者

基本上就这些。关键是理解“发布-订阅”思想,用接口隔离变化,再结合现代C++特性提升安全性和可用性。实际项目中也可考虑使用信号槽库(如Boost.Signals2)来简化实现。

以上就是c++++怎么实现一个观察者模式_c++事件通知与解耦机制实现的详细内容,更多请关注其它相关文章!


# 多路  # 营销推广流程视频教学  # 湛江网站推广之家电话  # 十堰企业网站推广收费  # 广州网站推广方案效果  # 遵化网站建设服务  # 盐城百度关键词推广营销  # seo权重叠加技术  # 长安区网站品牌推广  # 企业为什么网站建设不好  # 松原全网营销推广  # 是一种  # 复用  # go  # 管理机制  # 如何实现  # 移除  # 何为  # 是一个  # 都是  # 多线程  # red  # stream  # ios  # c++  # ai 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: 印象笔记如何设提醒任务防漏执行_印象笔记设提醒任务防漏执行【任务提醒】  在Qt QML中通过Python字典动态更新TextEdit内容的教程  J*aScript map 迭代中检测空数组元素的有效方法  Vue.js 图片显示异常排查:理解应用挂载范围与DOM ID唯一性  零跑汽车11月交付量达70327台 实现连续9个月正增长  晋江读书网页版在线登录 晋江读书电脑版官网  可靠CSGO开箱平台解析 CSGO开箱网合集  J*aScript中向JSON对象添加新属性的正确姿势  Yandex免登录官网入口_俄罗斯Yandex搜索引擎直达链接  PHP表单数据传递:如何通过隐藏输入字段获取动态ID  外媒分析《GTA6》定价:卖100美元可以但真没必要!  Win11怎么修改默认浏览器_Windows 11设置Chrome为默认  快手网页版在线登录 快手网页版官网入口快速访问  Mac终端命令大全_Mac常用Terminal指令速查  邮政快递单号查询入口 邮政快递物流信息在线查询入口  Win10如何清理注册表垃圾 Win10注册表维护与优化指南【慎用】  QQ邮箱官方网站登录入口_QQ邮箱网页版在线使用  字由网在线版登录地址 字由网网页版安全入口  手机屏幕碎了但能正常使用怎么办 手机外屏碎裂的修复建议  将HTML Canvas内容转换为可上传的图像文件(File对象)  响应式图片在网页设计中的正确实现方法  如何在复杂的电商平台中优雅地管理共享资源并确保正确重定向,使用spryker-shop/resource-share-page模块助你一臂之力  在J*a中如何在J*a中使用异常机制记录错误日志_异常日志实践经验  Typer应用中动态命令行参数的解析与处理  qq游戏大厅官方下载_qq游戏免费下载安装入口  PPT平滑切换怎么做 PPT炫酷“平滑”切换动画制作教程【必学】  mysql通配符支持数字匹配吗_mysql通配符能否用于数字匹配的解析  特斯拉自动驾驶房车计划曝光 原型车将于2027年亮相  J*a TimerTask中HashMap意外清空的深层原因与解决方案  Win10怎么设置静态IP地址 Win10手动配置IP地址步骤【指南】  CSS Flexbox与媒体查询:实现响应式布局中元素的并排与堆叠  多闪网页版在线观看免费入口_多闪官网访问入口  Composer中的^和~符号代表什么_精通Composer版本号语义化约束  Win10怎么制作U盘启动盘 Win10系统安装U盘制作教程【详解】  动漫花园资源网使用步骤_动漫花园资源网下载流程  J*a如何使用AtomicInteger控制计数_J*a无锁计数器性能分析  如何使用Rector自动化升级旧代码_通过Composer安装和配置Rector进行代码重构  在WordPress中通过REST API获取BasicAuth保护的远程文章  Win11文件资源管理器卡顿怎么修 Win11重置资源管理器进程优化响应速度【修复方法】  J*aScript Promise链中如何正确终止后续.then执行并处理错误  基于动态规划的房屋花卉种植最小成本算法详解  纯CSS与HTML网格布局的HTML精简策略:SVG与JS方案解析  不同用户不同价格! 索尼开启账户个性化定价测试  利用Bokeh CustomJS动态控制DataTable列可见性  韩剧圈正版入口页面_韩剧圈官网登录链接  Pandas DataFrame:高效添加条件计算列  微信群消息显示延迟如何解决 微信群消息刷新优化方法  Golang如何测试channel通信行为_Golang channel通信测试与分析方法  mysql备份恢复性能优化_mysql备份恢复性能优化方法  Win11截图该按哪些键 Win11截屏完整流程解析【教程】 

搜索