首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
C++实现轻量级观察者模式(RAII版) _ 结合std::function的回调管理【源码】

C++实现轻量级观察者模式(RAII版) _ 结合std::function的回调管理【源码】

热心网友
14
转载
2026-04-29

C++实现轻量级观察者模式(RAII版) _ 结合std::function的回调管理【源码】

C++实现轻量级观察者模式(RAII版) _ 结合std::function的回调管理【源码】

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

为什么 std::function + RAII 是观察者注销最稳妥的组合

在C++里手动管理观察者的生命周期,就像走钢丝——稍有不慎,悬空回调(dangling callback)就会导致程序崩溃。问题的核心在于,std::function本身只是个可调用对象的包装器,它并不持有被调用对象的所有权。这就引出了一个关键的设计原则:订阅行为必须返回一个可析构的句柄,让析构动作等同于自动退订。这套机制,远比传统的“注册时传入裸指针,再手动调用unregister”要可靠得多。

想想看,一个典型的翻车现场是什么样?std::function捕获了this指针,可被观察的对象早就销毁了,观察者列表里却还躺着一个试图调用已释放内存的函数对象,崩溃只是时间问题。

要避免这种局面,有几个要点必须把握:

  • 生命周期管理是前提:如果回调需要访问被观察对象的状态,那么必须使用std::shared_ptrstd::weak_ptr来管理其生命周期。
  • 句柄必须封装:句柄类型绝不能是裸指针或引用。更推荐的做法是,用一个轻量级结构体去封装std::list<...>::iterator,或者使用原子计数器生成的唯一ID。
  • 警惕迭代器失效:务必避免在回调执行过程中去修改观察者容器(比如一边遍历一边删除),否则迭代器失效会引发未定义行为。
最稳妥的组合是因RAII句柄在析构时自动退订,避免悬空回调;std::function需配合shared_ptr或weak_ptr管理被观察对象生命周期,防止use-after-free。

如何设计可自动注销的订阅句柄(RAII 核心)

所谓RAII句柄,其本质就是一个与作用域绑定的“退订令牌”。典型的实现思路是,让subscribe()函数返回一个ObserverHandle对象,而这个对象的析构函数,会默默地执行内部的unsubscribe()逻辑。

来看一个关键的结构示例:

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

struct ObserverHandle {
    std::list>* list_ = nullptr;
    std::list>::iterator it_;

    ObserverHandle(std::list>& l,
                   std::list>::iterator i)
        : list_(&l), it_(i) {}

    ~ObserverHandle() {
        if (list_) list_->erase(it_);
    }

    ObserverHandle(const ObserverHandle&) = delete;
    ObserverHandle& operator=(const ObserverHandle&) = delete;
};

这里有两点需要特别注意:首先,list_存储的是原始指针,因为句柄并不拥有这个容器,它只是需要知道在哪里执行退订操作。其次,迭代器it_必须在构造时立即捕获并保存,延迟获取是行不通的。

std::function 回调捕获方式对生命周期的影响

回调能否安全执行,完全取决于其捕获方式是否与被观察者或观察者对象的生命周期同步。错误的捕获,直接通向use-after-free的深渊:

  • [this]{ ... }高危操作!如果this所指的对象先于观察者容器被销毁,那么触发回调就是未定义行为。
  • [ptr = shared_from_this()]{ ... }:安全,但前提是被观察的类需要继承自std::enable_shared_from_this
  • [w = weak_ptr_to_observer]{ ... }:适用于观察者是独立对象的情况,在回调执行前先用w.lock()判断对象是否存活。
  • 纯函数对象或静态函数:最为轻量,但代价是无法访问任何实例状态。

性能上也有个小提示:捕获std::shared_ptr会带来原子计数的开销。对于高频触发的事件,建议使用std::weak_ptr配合lock()判空,这样可以避免强引用导致的循环引用问题。

线程安全边界在哪?别指望 RAII 句柄自动解决并发问题

这是一个常见的误解:用了RAII句柄就等于线程安全。事实上,ObserverHandle自身的析构操作(操作它自己持有的迭代器和指针)可以是线程安全的,但观察者容器本身的读写操作,默认并不是线程安全的

具体的加锁策略,需要根据使用场景来决定:

  • 单线程环境:最简单,无需加锁,std::list配合RAII句柄完全足够。
  • 多生产者/单消费者(例如主线程发布通知,工作线程订阅):在向容器写入(订阅/退订)时需要加std::mutex锁;而在读取容器遍历通知时,可以尝试无锁,但必须确保遍历期间容器结构不被修改。
  • 高频多线程通知:可以考虑使用std::vector>配合原子索引快照的策略,来彻底规避迭代器失效的问题。

最后,一个真正容易被忽略的细节是:即使使用了RAII句柄,如果两个线程同时调用subscribe()函数,你仍然需要保护容器的插入操作——因为句柄只负责管理“退订”的生命周期,它可管不了“注册”时的并发安全。

来源:https://www.php.cn/faq/2388403.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

C++实现轻量级观察者模式(RAII版) _ 结合std::function的回调管理【源码】
编程语言
C++实现轻量级观察者模式(RAII版) _ 结合std::function的回调管理【源码】

C++实现轻量级观察者模式(RAII版) _ 结合std::function的回调管理【源码】 为什么 std::function + RAII 是观察者注销最稳妥的组合 在C++里手动管理观察者的生命周期,就像走钢丝——稍有不慎,悬空回调(dangling callback)就会导致程序崩溃。问题

热心网友
04.29
C++ std::views::filter过滤容器数据 _ C++20管道操作符应用【详解】
编程语言
C++ std::views::filter过滤容器数据 _ C++20管道操作符应用【详解】

std::views::filter 返回惰性视图,不持有数据、不支持随机访问、无 size(),故不能直接用[]或传给需 vector 的函数;须转容器、用算法或谨慎处理捕获生命周期。 直接把 std::views::filter 的结果当容器用,是很多开发者上手 C++20 范围库时踩的第一个坑

热心网友
04.28
C++实现动态库DLL加载的包装类 _ RAII管理加载与导出函数【源码】
编程语言
C++实现动态库DLL加载的包装类 _ RAII管理加载与导出函数【源码】

RAII封装动态库加载需确保HMODULE生命周期与对象绑定:构造时调用LoadLibrary并校验非空,析构时仅对非空句柄调用FreeLibrary;GetProcAddress应延迟至每次调用前执行并检查句柄有效性,避免缓存失效指针。 如何用 RAII 封装 LoadLibrary 和 GetP

热心网友
04.28
C++ std::all_of与any_of案例演示 _ 容器条件快速检索的高效方法【详解】
编程语言
C++ std::all_of与any_of案例演示 _ 容器条件快速检索的高效方法【详解】

空容器上调用 std::all_of 返回 true 是标准定义的空真,表示“无反例”而非“非空且满足”;正确校验需显式合取 !v empty() && std::all_of( ),且前者须前置。 std::all_of 空容器返回 true 是设计,不是 bug 开门见山,先说一个让不少开发

热心网友
04.28
C++如何获取硬盘分区的详细挂载信息 _ filesystem库实战【实战】
编程语言
C++如何获取硬盘分区的详细挂载信息 _ filesystem库实战【实战】

C++如何获取硬盘分区的详细挂载信息 _ filesystem库实战【实战】 std::filesystem::space() 能不能拿到挂载点路径? 答案是:不能。很多开发者会误以为std::filesystem::space()能提供完整的磁盘信息,其实它只负责一件事:返回指定路径所在文件系统的

热心网友
04.28

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

HDFS配置怎样提升集群的稳定性
编程语言
HDFS配置怎样提升集群的稳定性

要提升HDFS集群的稳定性,这些配置与优化思路值得关注 想让你的Hadoop分布式文件系统(HDFS)集群运行得更稳定、更可靠吗?这既是一项系统工程,也有一套清晰的优化路径——关键在于,你是否在硬件选型、参数配置、运维管理等核心层面都进行了系统性的规划与调优。下面这张图,可以帮助你快速建立起一个关于

热心网友
04.29
HDFS配置里如何调整数据块的副本策略
编程语言
HDFS配置里如何调整数据块的副本策略

HDFS副本策略调整指南 一 核心概念与层级 要玩转HDFS的副本策略,得先理清几个核心概念。它们像齿轮一样层层咬合,共同决定了数据最终落在哪里。 副本因子:这个最好理解,就是一个数据块要存几份。它直接决定了数据的可靠性和存储开销,默认值是3,算是可靠性与成本之间的经典平衡点。 副本放置策略:这是N

热心网友
04.29
HDFS配置怎样实现数据的容错
编程语言
HDFS配置怎样实现数据的容错

HDFS:一个为容错而生的分布式文件系统 在分布式存储领域,数据的安全性与可靠性是系统设计的核心。HDFS(Hadoop分布式文件系统)之所以能成为大数据生态的基石,关键在于其设计了一套多层次、自动化的容错机制。这套机制确保了在硬件故障、网络异常等常见问题发生时,数据依然保持完整且服务持续可用。本文

热心网友
04.29
HDFS配置中如何设置合理的权限
编程语言
HDFS配置中如何设置合理的权限

在HDFS中设置合理权限:一份实战指南 在Hadoop分布式文件系统(HDFS)中,权限管理绝非小事。它直接关系到数据的安全底线和系统的稳定运行。那么,如何为HDFS中的文件和目录设置一套既安全又实用的权限规则呢?下面这份指南,或许能给你带来清晰的思路。 1 基本概念 在动手之前,先得理清几个核心

热心网友
04.29
HDFS配置里如何实现数据压缩
编程语言
HDFS配置里如何实现数据压缩

在Hadoop分布式文件系统(HDFS)中实现数据压缩 处理海量数据时,存储成本与传输效率是两大核心挑战。HDFS提供了多种数据压缩方案,能够有效降低存储空间占用并提升数据处理性能。本文将详细介绍在HDFS中启用和配置数据压缩的几种实用方法。 1 配置文件设置 最直接且全局生效的方式是通过修改Ha

热心网友
04.29