C++那些事》之enable_shared_from_this

《C++那些事》之enable_shared_from_this

std::enable_shared_from_this它被用作需要创建与现有对象共享所有权的 std::shared_ptr 实例的类的基类。

使用 enable_shared_from_this 的类必须继承自 enable_shared_from_this

当一个类 A 继承了 std::enable_shared_from_this<A>,并在某个地方创建了一个 std::shared_ptr<A> 的智能指针时,通过调用 shared_from_this 函数可以得到一个指向该对象的 std::shared_ptr,且该 std::shared_ptr 与原始的 std::shared_ptr 共享对象的所有权。

例如:

class Foo
    : public std::enable_shared_from_this<Foo> 
{

}

简单实现如下

std::enable_shared_from_this 是一个模板类,定义如下:

template<class T>
class enable_shared_from_this
{
protected:
    constexpr enable_shared_from_this() noexcept;
    enable_shared_from_this(const enable_shared_from_this&) noexcept;
    enable_shared_from_this& operator=(const enable_shared_from_this&) noexcept;
    ~enable_shared_from_this();
public:
    shared_ptr<T> shared_from_this();
    shared_ptr<const T> shared_from_this() const;
};

enable_shared_from_this 模板类的实现依赖于两个关键类,即 std::shared_ptrstd::weak_ptr。它的核心思想是通过将 std::shared_ptr 对象作为类成员变量保存,并在需要共享该对象所有权的地方返回一个指向该对象的 std::shared_ptr

enable_shared_from_this 类的构造函数中,将 weak_this 成员变量初始化为一个空的 std::weak_ptr。该成员变量用于保存指向该类的 std::shared_ptr 对象,该对象是通过调用 shared_from_this 函数返回的。

标准库的std::enable_shared_from_this 的实现大致如下:

template<typename _Tp>
    class enable_shared_from_this
    {
    protected:
      constexpr enable_shared_from_this() noexcept { }

      enable_shared_from_this(const enable_shared_from_this&) noexcept { }

      enable_shared_from_this&
      operator=(const enable_shared_from_this&) noexcept
      { return *this; }

      ~enable_shared_from_this() { }

    public:
      shared_ptr<_Tp>
      shared_from_this()
      { return shared_ptr<_Tp>(this->_M_weak_this); }

      shared_ptr<const _Tp>
      shared_from_this() const
      { return shared_ptr<const _Tp>(this->_M_weak_this); }

#if __cplusplus > 201402L || !defined(__STRICT_ANSI__) // c++1z or gnu++11
#define __cpp_lib_enable_shared_from_this 201603
      weak_ptr<_Tp>
      weak_from_this() noexcept
      { return this->_M_weak_this; }

      weak_ptr<const _Tp>
      weak_from_this() const noexcept
      { return this->_M_weak_this; }
#endif

    private:
      template<typename _Tp1>
 void
 _M_weak_assign(_Tp1* __p, const __shared_count<>& __n) const noexcept
 { _M_weak_this._M_assign(__p, __n); }

      // Found by ADL when this is an associated class.
      friend const enable_shared_from_this*
      __enable_shared_from_this_base(const __shared_count<>&,
         const enable_shared_from_this* __p)
      { return __p; }

      template<typename, _Lock_policy>
 friend class __shared_ptr;

      mutable weak_ptr<_Tp>  _M_weak_this;
    };

在构造函数中,weak_this_ 被初始化为一个空的 std::weak_ptr。在使用 shared_from_this 函数时,它将返回一个指向 this 对象的 std::shared_ptr,如果此时没有任何 std::shared_ptr 对象管理该对象,将会抛出 std::bad_weak_ptr 异常。

需要注意的是,shared_from_this 函数必须在至少存在一个 std::shared_ptr 对象管理该对象时才能调用,否则将会出现未定义的行为。同时,shared_from_this 函数也不能在对象的构造函数中被调用,因为此时还没有任何 std::shared_ptr 对象管理该对象。

M_weak_this_ 是一个 std::weak_ptr 对象,它保存的是指向当前对象的一个弱引用(weak reference),而不是强引用(shared reference)。

std::enable_shared_from_this 的实现中,wM_eak_this_ 被初始化为一个空的 std::weak_ptr,因为此时还没有任何 std::shared_ptr 对象管理当前对象,所以也就无法通过任何 std::shared_ptr 对象来创建一个指向当前对象的 std::shared_ptr

当我们在某个地方创建了一个 std::shared_ptr 对象来管理当前对象时,我们需要在这个 std::shared_ptr 对象创建之后,调用 std::enable_shared_from_this 中提供的 shared_from_this 函数来获得一个指向当前对象的 std::shared_ptr。在这个过程中,shared_from_this 函数将使用 _M_weak_this 中保存的弱引用来创建一个指向当前对象的 std::shared_ptr

需要注意的是,如果在任何一个时刻,没有任何 std::shared_ptr 对象管理当前对象,那么 _M_weak_this 中保存的弱引用就会被销毁,此时再尝试通过 shared_from_this 函数来创建一个指向当前对象的 std::shared_ptr,将会抛出 std::bad_weak_ptr 异常。因此,使用 std::enable_shared_from_this 时需要特别注意对象的生命周期和 shared_ptr 的管理方式。

enable_shared_from_this可以使用是在shared_ptr构造过程中,有相应的设置过程。

shared_ptr调用构造__shared_ptr__shared_ptr构造会调用执行_M_enable_shared_from_this_with_M_enable_shared_from_this_with调用执行__base->_M_weak_assign的过程,此时会构造_M_weak_this

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++智能指针shared_ptr讲解与使⽤ ⼿动管理的弊端 在简单的程序中,我们不⼤可能忘记释放 new 出来的指针,但是随着程序规模的增⼤,我们忘了 delete 的概率也随之增⼤。在 C++ 中 new 出来的指针,赋值意味着引⽤的传递,当赋值运算符同时展现出"值拷贝"和"引⽤传递"两种截然不同的语义时,就很容易导致"内 存泄漏"。 ⼿动管理内存带来的更严重的问题是,内存究竟要由谁来分配和释放呢?指针的赋值将同⼀对象的引⽤散播到程序各处,但是该对象的释放 却只能发⽣⼀次。当在代码中⽤完了⼀个资源指针,该不该释放 delete 掉它?这个资源极有可能同时被多个对象拥有着,⽽这些对象中的 任何⼀个都有可能在之后使⽤该资源,其余指向这个对象的指针就变成了"野指针";那如果不 delete 呢?也许你就是这个资源指针的唯 ⼀使⽤者,如果你⽤完不 delete,内存就泄漏了。 资源的拥有者是系统,当我们需要时便向系统申请资源,当我们不需要时就让系统⾃⼰收回去(Garbage Collection)。当我们⾃⼰处理的时 候,就容易出现各种各样的问题。 C++中的智能指针 ⾃C++11起,C++标准提供两⼤类型的智能指针: 1. Class shared_ptr实现共享式拥有(shared ownership)概念。多个智能指针可以指向相同对象,该对象和其相关资源会在"最后⼀个引 ⽤(reference)被销毁"时候释放。为了在结构复杂的情境中执⾏上述⼯作,标准库提供了weak_ptr、bad_weak_ptr和 enable_shared_from_this等辅助类。 2. Class unique_ptr实现独占式拥有(exclusive ownership)或严格拥有(strict ownership)概念,保证同⼀时间内只有⼀个智能指针 可以指向该对象。它对于避免资源泄露(resourece leak)——例如"以new创建对象后因为发⽣异常⽽忘记调⽤delete"——特别有⽤。 注:C++98中的Class auto_ptr在C++11中已不再建议使⽤。 share_ptr 智能指针是(⼏乎总是)模板类,shared_ptr 同样是模板类,所以在创建 shared_ptr 时需要指定其指向的类型。shared_ptr 负责在不使 ⽤实例时释放由它管理的对象,同时它可以⾃由的共享它指向的对象。 shared_ptr 使⽤经典的 "引⽤计数" 的⽅法来管理对象资源。引⽤计数指的是,所有管理同⼀个裸指针( raw pointer )的 shared_ptr,都共享⼀个引⽤计数器,每当⼀个 shared_ptr 被赋值(或拷贝构造)给其它 shared_ptr 时,这个共享的引⽤计数器就加 1,当⼀个 shared_ptr 析构或者被⽤于管理其它裸指针时,这个引⽤计数器就减1,如果此时发现引⽤计数器为0,那么说明它是管理这个 指针的最后⼀个 shared_ptr 了,于是我们释放指针指向的资源。 在底层实现中,这个引⽤计数器保存在某个内部类型⾥(这个类型中还包含了 deleter,它控制了指针的释放策略,默认情况下就是普通的 delete操作),⽽这个内部类型对象在 shared_ptr 第⼀次构造时以指针的形式保存在 shared_ptr 中(所以⼀个智能指针的析构会影响到 其他指向同⼀位置的智能指针)。shared_ptr 重载了赋值运算符,在赋值和拷贝构造另⼀个 shared_ptr 时,这个指针被另⼀个 shared_ptr 共享。在引⽤计数归零时,这个内部类型指针与 shared_ptr 管理的资源⼀起被释放。此外,为了保证线程安全性,引⽤计数 器的加1,减1操作都是 原⼦操作,它保证 shared_ptr 由多个线程共享时不会爆掉。 对于 shared_ptr 在拷贝和赋值时的⾏为,《C++Primer第五版》中有详细的描述: 每个 shared_ptr 都有⼀个关联的计数值,通常称为引⽤计数。⽆论何时我们拷贝⼀个 shared_ptr,计数器都会递增。 例如,当⽤⼀个 shared_ptr 初始化另⼀个 shred_ptr,或将它当做参数传递给⼀个函数以及作为函数的返回值时,它所关联的计数 器就会递增。当我们给 shared_ptr 赋予⼀个新值或是 shared_ptr 被销毁(例如⼀个局部的 shared_ptr 离开其作⽤域)时,计数 器就会递减。⼀旦⼀个 shared_ptr 的计数器变为0,它就会⾃动释放⾃⼰所管理的对象。 下⾯看⼀个常见⽤法,包括: 1. 创建 shared_ptr 实例 2. 访问所指对象 3. 拷贝和赋值操作 4. 检查引⽤计数。 #include <iostream> #include

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值