当需要在类内部安全获取指向当前对象的std::shared_ptr时应使用std::enable_shared_from_this,因为直接使用std::shared_ptr(this)会创建独立的引用计数导致双重释放;正确做法是让类继承std::enable_shared_from_this并通过std::make_shared创建对象后调用shared_from_this(),从而确保所有返回的shared_ptr共享同一控制块,适用于回调、自引用等场景,且不可在构造函数中或未被shared_ptr管理时调用,否则会抛出异常或导致未定义行为。

当你需要在类的内部安全地获取指向当前对象的
std::shared_ptr
时,就应该使用
enable_shared_from_this
。
直接使用
std::make_shared(this)
或者
std::shared_ptr(this)
是错误且危险的,因为这会创建一个新的引用计数,导致多个独立的
shared_ptr
管理同一个对象,最终可能引发双重释放或未定义行为。
什么时候使用 enable_shared_from_this?
enable_shared_from_this
的核心用途是:在对象已经被
shared_ptr
管理的前提下,类内部需要生成指向自身的
shared_ptr
。
常见场景包括:
回调机制中传递当前对象的共享指针(如异步操作、事件处理)需要在成员函数中将
this
作为
shared_ptr
传给其他函数或对象实现“自引用”逻辑,比如对象需要把自己加入某个共享容器
正确使用方式
要使用
enable_shared_from_this
,类必须:
继承
std::enable_shared_from_this
对象必须通过
std::shared_ptr
创建使用
shared_from_this()
获取
shared_ptr
#include #include struct MyClass : std::enable_shared_from_this { void do_something() { // 安全地获取指向自身的 shared_ptr auto self = shared_from_this(); some_other_function(self); } void some_other_function(std::shared_ptr ptr) { std::cout << "Called with shared_ptrn"; }};int main() { auto obj = std::make_shared(); // 必须用 shared_ptr 创建 obj->do_something(); // 此时 shared_from_this() 才能安全调用}
常见错误和注意事项
❌ 不要直接
new
对象然后调用
shared_from_this()
,会抛出
std::bad_weak_ptr
异常❌ 不要在构造函数中调用
shared_from_this()
,此时对象尚未被
shared_ptr
完全接管✅ 确保对象生命周期由
shared_ptr
管理✅ 多次调用
shared_from_this()
返回的
shared_ptr
共享同一引用计数
为什么不能直接 new 出来再用 shared_ptr 包装?
MyClass* p = new MyClass;std::shared_ptr ptr1(p);std::shared_ptr ptr2(p); // 危险!两个独立的 shared_ptr 管理同一对象
这样会造成两个独立的控制块,引用计数互不相干,析构时会崩溃。
而
enable_shared_from_this
内部使用
weak_ptr
跟踪对象是否已被
shared_ptr
管理,确保所有
shared_from_this()
返回的指针共享同一个控制块。
基本上就这些。只要记住:想在类内部返回
this
的
shared_ptr
,且对象是用
shared_ptr
管理的,就必须用
enable_shared_from_this
。
以上就是enable_shared_from_this何时使用 获取this的shared_ptr方法的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1470822.html
微信扫一扫
支付宝扫一扫