0

0

c++怎么实现观察者模式_c++事件触发机制【实战】

冰火之心

冰火之心

发布时间:2026-03-16 19:47:30

|

694人浏览过

|

来源于php中文网

原创

应使用 std::shared_ptr 管理观察者生命周期,被观察者以 std::vector<std::weak_ptr<ObserverBase>> 存储回调,每次 notify 前调用 lock() 检查有效性,避免野指针崩溃;同时需破除 lambda 捕获 this 导致的循环引用,并在多线程下通过快照机制防止迭代器失效。

c++怎么实现观察者模式_c++事件触发机制【实战】

std::function + std::vector 怎么存回调而不崩溃

直接裸存 std::function 没问题,但常见崩溃是因为观察者对象提前析构,而被观察者还拿着它的回调——调用时访问了野指针。

解决思路不是“怎么存”,而是“怎么安全地存”。推荐用 std::shared_ptr 管理观察者生命周期,并配合 std::weak_ptr 在被观察者中持有弱引用:

  • 观察者继承一个基类(如 ObserverBase),内部持有一个 std::shared_ptr<ObserverBase> 自引用
  • 被观察者用 std::vector<std::weak_ptr<ObserverBase>> 存储,每次触发前先 lock() 检查是否还活着
  • 避免用裸指针或 std::unique_ptr:前者不防析构,后者无法共享所有权

示例关键片段:

void Subject::notify() {
    for (auto& wptr : observers_) {
        if (auto ptr = wptr.lock()) {
            ptr->onEvent(data_);
        }
    }
}

为什么不用 Qt 的 signals/slots 做跨模块解耦

Qt 的 signal/slot 机制确实开箱即用,但它把观察者模式和 GUI 框架强绑定了。如果你的模块要跑在无 Qt 环境(比如嵌入式、服务端计算模块、测试 mock),或者需要和 Python/C 绑定层交互,就会卡住。

立即学习C++免费学习笔记(深入)”;

更实际的问题是信号连接的隐式生命周期管理容易出错:

  • connect(obj1, &A::sig, obj2, &B::slot) 默认是 Qt::AutoConnection,跨线程时可能变成队列传递,导致回调延迟或重入
  • 如果 obj2delete 但没显式 disconnect,下次发信号就 crash
  • 编译期不检查参数类型匹配,错误只在运行时报 QObject::connect: Cannot queue arguments of type 'XXX'

纯 C++ 实现反而更容易控制连接/断连时机,也方便加日志、统计、调试钩子。

‎ Gemini Storybook
‎ Gemini Storybook

Google Gemini推出的AI绘本生成工具

下载

std::function 捕获 lambda 导致内存泄漏怎么办

这是高频坑:写 observers_.emplace_back([this]() { this->handle(); });,结果 this 是被观察者自己——形成循环引用,谁都不会析构。

根本原因是捕获了被观察者,又把它存进被观察者的容器里。破环方法很简单:

  • 改用非捕获 lambda:[&]() { handle(); }(但仅限函数内联调用,不能存)
  • 改用普通函数指针或静态成员函数(无状态)
  • 最通用:让观察者自己提供回调,由它管理捕获内容,被观察者只存 std::function<void()>,不参与捕获逻辑

例如观察者暴露一个 getCallback() 接口,返回已绑定好上下文的 std::function,被观察者只负责调用,不构造。

多线程下 notify() 怎么避免迭代器失效和重复通知

一边在 notify() 遍历 std::vector,一边其他线程调用 register()/unregister() —— 迭代器立刻失效,push_backerase 都可能触发 realloc。

别用读写锁硬扛,性能差还容易死锁。更轻量的做法是:

  • 注册/注销操作走原子队列(如 std::queue<std::shared_ptr<ObserverBase>> + std::mutex),只在 notify() 开始前快照一次当前有效观察者列表
  • std::vector<std::weak_ptr<...>> + reserve() 预分配,避免 notify 中 realloc
  • 禁止在回调里调用 register/unregister;如需动态调整,改为 post 到事件队列异步处理

真正麻烦的不是并发本身,而是回调里再触发新事件——容易形成不可控的调用链。这点比线程安全更值得盯紧。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
视频后缀名都有哪些
视频后缀名都有哪些

视频后缀名都有avi、mpg、mpeg、rm、rmvb、flv、wmv、mov、mkv、ASF、M1V、M2V、MPE、QT、VOB、RA、RMJ、RMS、RAM、等等。更多关于视频后缀名的相关知识,详情请看本专题下面的文章,php中文网欢迎大家前来学习。

3895

2023.10.31

C++ Qt图形开发
C++ Qt图形开发

本专题专注于 C++ Qt框架在图形界面开发中的应用,系统讲解窗口设计、信号与槽机制、界面布局、事件处理、数据库连接与跨平台打包等核心技能,通过多个桌面应用项目实战,帮助学员快速掌握 Qt 框架并独立完成跨平台GUI软件的开发。

76

2025.08.15

C++ 图形界面开发基础(Qt方向)
C++ 图形界面开发基础(Qt方向)

本专题系统讲解 使用 C++ 与 Qt 进行图形界面(GUI)开发的核心技能,内容涵盖 Qt 项目结构、窗口组件、信号与槽机制、事件处理、布局管理、资源管理,以及跨平台编译与打包流程。通过多个小型桌面应用实战案例,帮助学习者掌握从界面设计到功能实现的完整 GUI 开发能力。

112

2025.12.05

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

188

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

135

2025.11.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

215

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

193

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

62

2026.01.05

chatgpt使用指南
chatgpt使用指南

本专题整合了chatgpt使用教程、新手使用说明等等相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11.5万人学习

C 教程
C 教程

共75课时 | 5.5万人学习

C++教程
C++教程

共115课时 | 22.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号