信息发布→ 登录 注册 退出

C++如何使用std::shared_from_this_C++智能指针管理与shared_from_this应用

发布时间:2025-11-17

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

在C++中,使用std::shared_ptr管理对象生命周期时,有时需要在类的成员函数内部返回当前对象的std::shared_ptr。直接通过newthis构造std::shared_ptr是危险且错误的做法,会导致多次释放或引用计数不一致。为了解决这个问题,C++提供了std::enable_shared_from_thisshared_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

以下是一个典型用法示例:

#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_ptrstd::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,就能安全使用。

标签:# 继承  # 可通过  # 几种  # 可以通过  # 要在  # 是由  # 多个  # 就能  # 抛出  # 已被  # 是一个  # this  # 事件  # 对象  # 处理器  # 指针  # 析构函数  # 构造函数  # 成员函数  # red  # 作用域  # stream  # ios  # c++  # ai  #   # 回调函数  
在线客服
服务热线

服务热线

4008888355

微信咨询
二维码
返回顶部
×二维码

截屏,微信识别二维码

打开微信

微信号已复制,请打开微信添加咨询详情!