智能指针——shared_ptr的原理及仿写

shared_ptr的原理及仿写

在这里插入图片描述

  • 共享指针允许多个指针指向同一份数据,因为它使用了引用计数,每多一个指针指向这个数据,引用技术加一,每销毁一个指针,引用技术减一,如果引用计数为0,则delete这个数据。
  • 但是共享指针也不能将同一个裸指针赋值给多个智能指针,因为这样会是两个独立的共享指针,它们会分别计数,也就是意味着到时候它们会重复释放。
  • 第一个没有计数器设计的版本:
template<typename T>
class MyShared_ptr {
private:
    //内部指针
    T* m_ptr;
    //计数器
    long* m_count;
public:
    //以裸指针构造共享指针,默认为nullptr
    //禁止隐式构造
    explicit MyShared_ptr(T* ptr = nullptr) : m_ptr(ptr) {
        //计数器初始化为1
        m_count = new long(1);
    }
    //以MyUnique_ptr右值构造共享指针
    MyShared_ptr(MyUnique_ptr<T>&& src) {
        //销毁独占指针,取出裸指针
        m_ptr = src.release();
        //计数器初始化为1
        m_count = new long(1);
    }
    //拷贝构造函数
    MyShared_ptr(const MyShared_ptr& src) {
        m_ptr = src.m_ptr;
        //拷贝计数器指针
        m_count = src.m_count;
        //计数器加一
        (*m_count)++;
    }
    //拷贝赋值
    MyShared_ptr& operator=(const MyShared_ptr& src) {
        if (&src == this) return *this;
        //如果自身不为空,先调用析构函数
        if (m_ptr != nullptr) {
            this->~MyShared_ptr();
        }
        m_ptr = src.m_ptr;
        m_count = src.m_count;
        (*m_count)++;
        return *this;
    }
    //以MyUnique_ptr右值 拷贝赋值
    MyShared_ptr& operator=(const MyUnique_ptr<T>&& src) {
        if (m_ptr != nullptr) {
            this->~MyShared_ptr();
        }
        //销毁独占指针,取出裸指针
        m_ptr = src.release();
        m_count = new long(1);
        return *this;
    }
    //析构函数
    ~MyShared_ptr() {
        //计数器减一
        (*m_count)--;
        //如果计数器为0,则释放指针空间和计数器空间
        if (0 == *m_count) {
            delete m_ptr;
            delete m_count;
            m_ptr = nullptr;
            m_count = nullptr;
        }
    }
    //获得内部指针地址
    T* get() {
        return m_ptr;
    }
    //重置指针
    void reset(T* ptr = nullptr) {
        //先调用析构函数
        this->~MyShared_ptr();
        m_ptr = ptr;
        m_count = new long(1);
    }
    //返回当前引用计数
    long use_count() {
        return *m_count;
    }
    //转换函数 bool
    operator bool() const {
        return m_ptr != nullptr;
    }
    //当前共享指针是否唯一
    bool unique() {
        if (1 == *m_count) return true;
        else return false;
    }
    //成员函数交换指针和计数器
    void swap(MyShared_ptr<T>& src) {
        T* temp_ptr;
        long* temp_count;
        temp_ptr = src.m_ptr;
        temp_count = src.m_count;
        src.m_ptr = m_ptr;
        src.m_count = m_count;
        m_ptr = temp_ptr;
        m_count = temp_count;
    }
    //重载*
    T& operator*() const {
        return *m_ptr;
    }
    //重载->
    T* operator->() const {
        return (&**this);
    }
};
//全局函数,有右值引用make_MyShared
template<class T>
MyShared_ptr<T> make_MyShared(MyShared_ptr<T>&& src) {
    return MyShared_ptr<T>(forward<MyShared_ptr<T>>(src));
}
  • 有计数器设计的版本
//计数器设计
template<typename T>
class MyCount {
private:
    //分别存储引用计数和弱指针计数
    long use_count;
    long weak_count;
public:
    //计数器的初始化
    MyCount(long u,long w) : use_count(u), weak_count(w) {}
    //弱指针计数++
    void Increase_weak_count() {
        ++weak_count;
    }
    //弱指针计数--
    void Decrease_weak_count() {
        --weak_count;
    }
    //引用计数++
    void Increase_use_count() {
        ++use_count;
    }
    //引用计数--
    void Decrease_use_count() {
        --use_count;
    }
    //获得引用计数
    long get_use_count() {
        return use_count;
    }
    //释放传入对象的内存
    void DestroyPtr(T* ptr) {
        delete ptr;
    }
    //若没有弱指针,则释放计数器
    void DestroyThis() {
        if(0 == weak_count)
            delete this;
    }
};

template<typename T>
class MyShared_ptr {
private:
    //内部指针
    T* m_ptr;
    //计数器指针
    MyCount<T>* m_count;
public:
    //以裸指针构造共享指针
    explicit MyShared_ptr(T* ptr = nullptr) : m_ptr(ptr) {
        //初始化计数器为 引用1 弱指针0
        m_count = new MyCount<T>(1,0);
    }
    //以独占指针构造共享指针
    MyShared_ptr(MyUnique_ptr<T>&& src) {
        //销毁独占指针并取出裸指针
        m_ptr = src.release();
        //初始化计数器为 引用1 弱指针0
        m_count = new MyCount<T>(1, 0);
    }
    //拷贝构造
    MyShared_ptr(const MyShared_ptr& src) {
        m_ptr = src.m_ptr;
        m_count = src.m_count;
        //引用计数++
        m_count->Increase_use_count();
    }
    //拷贝赋值
    MyShared_ptr& operator=(const MyShared_ptr& src) {
        if (&src == this) return *this;
        if (m_ptr != nullptr) {
            this->~MyShared_ptr();
        }
        m_ptr = src.m_ptr;
        m_count = src.m_count;
        //引用计数++
        m_count->Increase_use_count();
        return *this;
    }
    //拷贝独占指针赋值
    MyShared_ptr& operator=(const MyUnique_ptr<T>&& src) {
        if (m_ptr != nullptr) {
            this->~MyShared_ptr();
        }
        m_ptr = src.release();
        m_count = new MyCount<T>(1, 0);
        return *this;
    }
    //析构函数
    ~MyShared_ptr() {
        //引用计数--
        m_count->Decrease_use_count();
        //如果引用计数为零
        if (0 == m_count->get_use_count()) {
            //释放指针
            m_count->DestroyPtr(m_ptr);
            //销毁计数器(计数器内部判断有无弱指针)
            m_count->DestroyThis();
            m_ptr = nullptr;
            m_count = nullptr;
        }
    }
    //获得内部指针地址
    T* get() {
        return m_ptr;
    }
    //重置共享指针为新指针
    void reset(T* ptr = nullptr) {
        this->~MyShared_ptr();
        m_ptr = ptr;
        m_count = new MyCount<T>(1, 0);
    }
    //返回引用计数
    long use_count() {
        return m_count->get_use_count();
    }
    //转换函数
    operator bool() const {
        return m_ptr != nullptr;
    }
    //是否唯一
    bool unique() {
        if (1 == m_count->get_use_count()) return true;
        else return false;
    }
    //交换指针和计数器
    void swap(MyShared_ptr<T>& src) {
        T* temp_ptr;
        MyCount<T>* temp_count;
        temp_ptr = src.m_ptr;
        temp_count = src.m_count;
        src.m_ptr = m_ptr;
        src.m_count = m_count;
        m_ptr = temp_ptr;
        m_count = temp_count;
    }
    //重载*
    T& operator*() const {
        return *m_ptr;
    }
    //重载->
    T* operator->() const {
        return (&**this);
    }
    //设置弱指针类为友元类
    template<typename T>
    friend class MyWeak_ptr;
};

//全局函数,有右值引用make_MyShared
template<class T>
MyShared_ptr<T> make_MyShared(MyShared_ptr<T>&& src) {
    return MyShared_ptr<T>(forward<MyShared_ptr<T>>(src));
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
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_ptrC++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
智能指针C++中用于管理动态分配的内存的一种机制。它们可以自动地在不再需要时释放内存,从而避免内存泄漏和悬挂指针的问题。 shared_ptr是一种引用计数智能指针,它可以跟踪有多少个shared_ptr指向同一个对象,并在没有引用时自动释放内存。当创建shared_ptr时,它会增加引用计数,当销毁或重置shared_ptr时,它会减少引用计数。只有当引用计数为0时,才会真正释放内存。\[1\]shared_ptr可以通过构造函数接受一个指向动态分配对象的指针来创建,也可以使用std::make_shared函数来创建。\[2\] unique_ptr是一种独占智能指针,它拥有对动态分配对象的唯一所有权。当unique_ptr被销毁时,它会自动释放内存。unique_ptr不能被复制,但可以通过std::move函数进行转移所有权。\[3\]unique_ptr可以通过构造函数接受一个指向动态分配对象的指针来创建。 weak_ptr是一种弱引用智能指针,它指向由shared_ptr管理的对象,但不会增加引用计数。weak_ptr可以用于解决shared_ptr的循环引用问题,因为它不会导致对象无法释放。\[1\]weak_ptr可以通过shared_ptr的构造函数来创建。 auto_ptrC++11之前的一种智能指针,它类似于unique_ptr,但有一些限制和问题。auto_ptr在复制时会转移所有权,这可能导致悬挂指针的问题。因此,auto_ptr已经被unique_ptr取代,不推荐使用。 总结来说,shared_ptr是引用计数智能指针,unique_ptr是独占智能指针,weak_ptr是弱引用智能指针,而auto_ptr是已经过时的智能指针。它们各自有不同的用途和特点,可以根据具体的需求选择使用。 #### 引用[.reference_title] - *1* *2* *3* [C++11 解决内存泄露问题的智能指针shared_ptr、unique_ptr、weak_ptr](https://blog.csdn.net/weixin_44120785/article/details/128714630)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值