std::enable_shared_from_this消除异步回调引发的野指针问题

本文介绍了如何利用std::enable_shared_from_this解决C++中异步回调可能导致的野指针问题。通过捕获shared_ptr或weak_ptr,可以确保在回调执行时对象仍处于有效状态。特别强调了使用weak_from_this时的注意事项,如尽早保存weak_ptr以及避免将其作为类成员。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

     std::enable_shared_from_this这个C++组件完美解决了异步回调发生时宿主对象销毁的问题。C++提供了lambda表达式和各种异步函数,std::thread,std::async或者其他框架api都提供了异步回调方法。特别是lambda在异步应用中实在是太方便了,比如发送一个http请求,接口可能是这样的api::get(url, [](response){ //解析response})。用起来方便,阅读起来也容易理解,但是同时也带来了问题,那就是回调回来的时候宿主类对象已经销毁了,但是大多数情况下我们再处理网络结果的时候需要用到宿主类对象的一些数据,这就造成了问题,宿主对象已经销毁了,他的成员数据也一起销毁了,继续调用只会导致无法预期的问题,所以这里我们有三种应对方法:

第一,宿主类销毁后,切断回调和宿主之前的关系.C++不提供这样的方法,需要自己设计,自己重新设置回调指针为空,调用回调函数的地方判断回调指针为空后就不调用。这种方法比较麻烦,需要手动设置,违反了简易自动化的原则。如果使用QT的话会比较方便,因为QT的信号槽机制会检测接收对象是否销毁。

第二,如果回调没有回来,延长宿主对象的生命周期,继承std::enable_shared_from_this可以做到这一点,那就是让lambda捕获宿主对象的shared_ptr指针(share_from_this()),这样在回调回来之前宿主对象不会销毁。但是问题也很明显,宿主对象的生命周期就依赖于这些回调被调用的时间,有些场景不合适。

第三, 是最好的方式,那就是检测对象是否销毁,如果销毁就不调用,如果没有继续执行,同样继承std::enable_shared_from_this可以做到这一点。让回调lambda捕获宿主对象的弱引用指针(weak_from_this()),在回调回来以后,检测weak_ptr是否可用,可用则获取shared_ptr保证宿主生命周期,然后执行其他方法。不可用直接返回。

#include <iostream>
#include <thread>
#include <functional>
#incl
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值