c++的std::shared_from_this有什么用 安全地获取this的shared_ptr【智能指针】

std::shared_from_this用于已由shared_ptr管理的对象安全获取共享引用计数的shared_ptr;直接new shared_ptr(this)会创建独立控制块导致双重析构;必须公有继承enable_shared_from_this,并确保对象已被shared_ptr持有后才能调用。

std::shared_from_this 的作用是让一个已由 std::shared_ptr 管理的对象,能在成员函数内部安全地获取指向自身的、与原始 shared_ptr 共享引用计数的另一个 shared_ptr

为什么不能直接 new 一个 shared_ptr(this)?

直接写 std::shared_ptr(this) 是危险的:它会创建一个**新的控制块**,和原来管理该对象的 shared_ptr 互不感知。结果就是同一对象被两个独立的引用计数器管理,析构两次或提前释放,导致未定义行为(崩溃、内存错误)。

必须继承 enable_shared_from_this

要使用 shared_from_this(),类必须公有继承 std::enable_shared_from_this

  • 这个基类提供了一个弱引用(weak_ptr)用于内部记录对象是否已被 shared_ptr 持有;
  • 它本身不管理生命周期,只协助派生类安全生成共享指针;
  • 继承后,shared_from_this() 才能正确返回和原始 shared_ptr 共享控制块的实例。

只能在对象已被 shared_ptr 管理后调用

shared_from_this() 不是万能钥匙 —— 它要求当前对象**必须已经由至少一个 std::shared_ptr 拥有**,否则抛出 std::bad_weak_ptr 异常。

常见错误场景:

  • 在构造函数里调用(此时外部 shared_ptr 还没创建完成);
  • 对象是栈上分配或裸指针 new 出来的,从未交给 shared_ptr 管理;
  • 原始 shared_ptr 已销毁,但对象还在(比如通过 this 指针访问到已释放内存)。

典型用途:回调、异步操作、自我持有

最常见于需要把 this 安全传递给异步任务或回调函数的场景,避免悬挂指针或重复释放:

  • 网络库中,连接对象在收到数据后启动异步读取,需确保回调执行时对象仍存活;
  • 定时器回调中捕获 shared_from_this(),保证定时器触发时对象没被销毁;
  • 观察者模式中,被观察对象需向观察者传递自身智能指针,而非裸指针。

不复杂但容易忽略:继承、先由 shared_ptr 构造、再调用 —— 三者缺一不可。