答案:std::enable_shared_from_this允许类通过shared_from_this安全返回指向自身的shared_ptr,前提是对象已被shared_ptr管理。

在C++中,使用std::shared_ptr管理对象生命周期时,有时需要在类的成员函数内部返回当前对象的std::shared_ptr。直接通过new或this构造std::shared_ptr是危险且错误的做法,会导致多次释放或引用计数不一致。为了解决这个问题,C++提供了std::enable_shared_from_this和shared_from_this()机制。
什么是 shared_from_this?
std::enable_shared_from_this是一个模板基类,继承它后,派生类可以通过调用shared_from_this()安全地获取指向自身的std::shared_ptr。这个机制确保所有指向该对象的shared_ptr共享同一份引用计数。
关键点:只有当对象已经被至少一个std::shared_ptr管理时,才能调用shared_from_this(),否则会抛出std::bad_weak_ptr异常。
如何正确使用 enable_shared_from_this
以下是一个典型用法示例:
立即学习“C++免费学习笔记(深入)”;
#include#include class MyClass : public std::enable_shared_from_this { public: MyClass() { std::cout << "MyClass constructed\n"; } ~MyClass() { std::cout << "MyClass destructed\n"; } // 安全返回当前对象的 shared_ptr std::shared_ptr get_shared() { return shared_from_this(); } void do_something() { std::cout << "Doing something...\n"; } }; int main() { auto ptr = std::make_shared (); // 必须通过 shared_ptr 构造 auto another_ptr = ptr->get_shared(); // 正确:引用计数+1 ptr->do_something(); another_ptr->do_something(); return 0; } // 两个 shared_ptr 离开作用域后,对象才被销毁
说明:
- 类必须继承
std::enable_shared_from_this - 对象必须通过
std::shared_ptr创建,不能是栈对象或裸指针直接构造 - 调用
shared_from_this()前,对象必须已被shared_ptr接管
常见错误与注意事项
以下几种情况会导致运行时错误或未定义行为:
-
在构造函数中调用
shared_from_this():此时weak_ptr还未被初始化,会抛出异常。 -
对栈上对象调用
shared_from_this():即使类继承了enable_shared_from_this,但对象不是由shared_ptr管理,调用将失败。 -
重复从同一个
this创建shared_ptr:std::shared_ptr多次调用会导致多个独立的引用计数,最终多次析构。(this)
正确做法始终是:
- 使用
std::make_shared创建对象() - 在需要返回自身共享指针的地方使用
shared_from_this() - 避免在构造函数或析构函数中调用
shared_from_this()
应用场景举例
这种机制常用于:
-
回调函数中传递自身:比如注册事件处理器时,需要把
shared_ptr传给外部系统,防止对象提前销毁。 -
实现工厂方法:在静态函数中创建对象并返回
shared_ptr,同时允许内部方法获取共享所有权。 -
父子对象关系管理:子对象需要持有父对象的
shared_ptr时,可通过shared_from_this()安全获取。
基本上就这些。只要记住:继承enable_shared_from_this、用make_shared创建、不在构造/析构中调用shared_from_this,就能安全使用。










