浅谈std::enable_shared_from_this

问题背景:

         用过enable_shared_from_this的基本都知道,要正常使用这个类,生成的对象必须由智能指针管理.那为何一定要用智能指针呢?网上很多,基本讲的都是因为:如果不用智能指针,作为基类的enable_shared_from_this<A>没有初始化,生成的对象没有计数器,那它又是如何与智能指针关联起来的呢?

代码:

书上的代码,引出问题点;没啥其它特别的用处

#include <iostream>
#include <memory>

using namespace std;
class A:public enable_shared_from_this<A>
{
public:
    A(){
        cout << "call constructor" << endl;
    }
    ~A(){
        cout << "call destructor" << endl;
    }
    shared_ptr<A> getSelf(){
        return shared_from_this();
    }
};

int main()
{
    shared_ptr<A> sp1{new A};//A a;
//    A a;
//    shared_ptr<A> sp1(&a);
    shared_ptr<A> sp2 = sp1->getSelf();//智能指针是为了管理堆上的内容;不是shared_ptr<A>会发生一异常
    cout << sp2.use_count() << endl;
    return 0;
}

相关知识点:

  • ADL(可参考C++ template这本书)
  • C++ 类及继承在内存中的布局
  • 友元类

原理解释:

注:ubuntu下不好对图片标记就直接弄了截图了

首先看enable_shared_from_this模板类的实现,在类的内部(680行)可以看到有__shared_ptr友元类,这个是shared_ptr这个智能指针的基类(ps:在shared_ptr_base.h里实现),就说明__shared_ptr可以使用enable_shared_from_this的函数,也就是671/676行这两个函数

图一

 在__shared_ptr类中跟踪__enable_shared_from_this_base这个函数,可以发现这个函数在1382行的函数中使用了(图二), 而1380行这个函数在基类__shared_ptr的构造函数中使用.

图二

 

图三

总结,当我们使用了shared_ptr,先调用它的基类__shared_ptr的构造函数,如果实现的类有继承的话,在这个构造函数中就已经关联了enable_shared_from_this(通过图二1380行的函数),没有则是在构造函数中使用图二1388行的函数.

参考书籍:

<<C++服务器开发精髓>>

<<C++ template>>

随手一记,这篇写得比较水~~~~~~~~

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`std::enable_shared_from_this` 是一个模板类,其目的是为了解决在一个对象中保存 shared_ptr 的问题。它是 C++11 引入的一个特性。 在使用 `std::shared_ptr` 时,我们往往需要在对象中保存一个 `std::shared_ptr` 的副本,这样才能确保对象在使用完毕后不会被提前销毁。但是这种方式会导致一些问题,比如我们无法防止用户直接使用裸指针来操作对象,从而导致对象被提前销毁等问题。 这时候,我们可以使用 `std::enable_shared_from_this` 来解决这些问题。具体而言,我们需要继承 `std::enable_shared_from_this`,然后在对象中使用 `shared_from_this()` 方法来获取一个指向当前对象的 `std::shared_ptr`。 下面是一个示例代码: ```c++ #include <iostream> #include <memory> class MyClass : public std::enable_shared_from_this<MyClass> { public: std::shared_ptr<MyClass> get_shared_ptr() { return shared_from_this(); } }; int main() { std::shared_ptr<MyClass> p(new MyClass); std::shared_ptr<MyClass> q = p->get_shared_ptr(); std::cout << "p.use_count() = " << p.use_count() << std::endl; std::cout << "q.use_count() = " << q.use_count() << std::endl; return 0; } ``` 在这个示例中,我们定义了一个名为 `MyClass` 的类,并且继承了 `std::enable_shared_from_this`。然后,我们在类中定义了一个名为 `get_shared_ptr()` 的方法,该方法使用了 `shared_from_this()` 方法来获取一个指向当前对象的 `std::shared_ptr`。在 `main()` 函数中,我们先创建了一个 `std::shared_ptr` 对象 `p`,然后通过 `p` 调用 `get_shared_ptr()` 方法获取了一个指向同一个对象的 `std::shared_ptr` 对象 `q`。最后,我们输出了 `p` 和 `q` 的引用计数,可以看到它们的引用计数都是 2。 需要注意的是,在使用 `std::enable_shared_from_this` 时,我们需要确保对象已经被一个 `std::shared_ptr` 管理,否则使用 `shared_from_this()` 方法会导致程序崩溃。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值