C++标准库学习之weak_ptr智能指针用法详解

 更新时间:2023年12月06日 14:22:53   作者:tsm  
这篇文章主要为大家详细介绍了C++标准库中weak_ptr智能指针用法的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下

在上一篇文章中介绍了 shared_ptr 这种智能指针,他为了解决在指针没有引用的情况下自动回收资源这种情况而产生的,但是在部分情况下会产生另一种问题,那就是两个智能指针被循环引用 , a 使用 shared_ptr 引用了b ,同时 b 使用 shared_ptr 引用了 a,在 方法执行完成后 ,a 与 b 的智能指针都会被 释放,但是 由于他们互相持有 , a 与 b 的 used_count >0 ,那么就不会释放他们的资源,看下面的例子

#include <iostream>
#include <memory>
#include <vector>
using namespace std;
class Person{
public:
    // 构造方法 ,name 是必要参数  mum 与 dad 是非必要参数 
    Person(string name,shared_ptr<Person> mum = nullptr ,shared_ptr<Person> dad = nullptr)
    :name(name) ,mum(mum),dad(dad)// 此种初始化方式被经常应用到 jni 源码上
    {
    }
    // 析构函数,资源被释放时会被调用
    ~Person(){
        cout<<"对象名称是->" << name<< "被回收了"<<endl;
    }
    // 向量 
    vector<shared_ptr<Person>> child;
    //名字
    string name;
    // 智能指针引用mum
    shared_ptr<Person> mum;
    //智能指针引用dad
    shared_ptr<Person> dad;
};
// 初始化 家人,让他们循环引用
shared_ptr<Person> getFamily(const string& name){
    shared_ptr<Person> mum(new Person(name +"'s Mum"));
    shared_ptr<Person> dad(new Person(name +"'s Dad"));
    shared_ptr<Person> child(new Person(name,mum,dad));
    mum->child.push_back(child);
    dad->child.push_back(child);
    return child;
}
int main(){
    shared_ptr<Person> tsm=  getFamily("tsm");
    cout<< "------------before  change-----------"<<endl;
    cout << tsm->dad->name<<endl;
    cout << tsm->mum->name<<endl;
    cout << tsm->name<<endl;
    cout<<"tsm 的引用个数" <<tsm.use_count()<<endl;
    cout<< "------------start change-----------"<<endl;
    tsm = getFamily("tsm1");
    cout<< "------------after change-----------"<<endl;
    system("pause");
    return 0;
}

结果:

D:\CWorkSpace\tsmTest\cmake-build-debug\tsmTest.exe
------------before  change-----------
tsm's Dad
tsm's Mum
tsm
tsm 的引用个数3
------------start change-----------
------------after change-----------
. . . pause . . . 

发现在为 tsm 这个变量二次赋值时,第一次 shared_ptr 指针指向的资源内存应该是被释放的,但是他们的析构函数并没有被执行,那就证明他的资源并没有被释放,那么我们如何来修改内,这就引出了我们今天的主角 weak_ptr

修改如下

class Person{
public:
    // 构造方法 ,name 是必要参数  mum 与 dad 是非必要参数
    Person(string name,shared_ptr<Person> mum = nullptr ,shared_ptr<Person> dad = nullptr)
    :name(name) ,mum(mum),dad(dad)// 此种初始化方式被经常应用到 jni 源码上
    {
    }
    // 析构函数,资源被释放时会被调用
    ~Person(){
        cout<<"对象名称是->" << name<< "被回收了"<<endl;
    }
    // 将向量中的引用使用  weak_ptr ,打破互相引用即可
    vector<weak_ptr<Person>> child;
    //名字
    string name;
    // 智能指针引用mum
    shared_ptr<Person> mum;
    //智能指针引用dad
    shared_ptr<Person> dad;
};

修改后的结果如下:

------------before  change-----------
tsm's Dad
tsm's Mum
tsm
tsm 的引用个数1
------------start change-----------
对象名称是->tsm被回收了
对象名称是->tsm's Dad被回收了
对象名称是->tsm's Mum被回收了
------------after change-----------

发现在使用 weak_ptr 打破互相引用的后, 被覆盖的 tsm 的析构函数被成功调用了,

weak_ptr 的使用场景主要有2个

1: 打破 shared_ptr 使用混乱出现的情况导致循环引用的情况,

2: 如果持有该 shared_ptr 的对象的生命周期是大于 该 shared_ptr 指针的生命周期的情况,就会导致 shared_ptr 资源无法被回收,

weak_ptr 的创建方式有3中

1:

weak_ptr<Person> t0; //构造空对象

2:

weak_ptr<Person> t1(t0); //使用拷贝构造函数

3:

shared_ptr<Person> tsm=  getFamily("tsm");

weak_ptr<Person> t2(tsm);

检查weak_ptr 是否还有效

t2.expired() 

expired 源码如下

bool
expired() const noexcept
{ return _M_refcount._M_get_use_count() == 0; }

可以看到 返回的是 use_count ,那就证明 if(!t2.expired) 则证明他是有效的

使用 weak_ptr,使用lock 方法将 weak_ptr 转换成 shared_ptr ,防止在使用过程中资源被回收

shared_ptr<Person> tt =  t2.lock();

看一下lock 的源码

shared_ptr<_Tp>
lock() const noexcept
{ return shared_ptr<_Tp>(*this, std::nothrow); }

到此这篇关于C++标准库学习之weak_ptr智能指针用法详解的文章就介绍到这了,更多相关C++ weak_ptr智能指针内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言 二叉树的链式存储实例

    C语言 二叉树的链式存储实例

    本篇文章主要介绍C语言中二叉树的链式存储,这里提供了一个实例代码进行参考,这样对二叉树的链式存储有更深入的了解,希望能帮到学习这块知识的同学
    2016-07-07
  • VS Code 中搭建 Qt 开发环境方案分享

    VS Code 中搭建 Qt 开发环境方案分享

    这篇文章主要介绍了VS Code 中搭建 Qt 开发环境方案分享的相关资料,需要的朋友可以参考下
    2022-12-12
  • C语言实现绘制南丁格尔玫瑰图的示例代码

    C语言实现绘制南丁格尔玫瑰图的示例代码

    玫瑰图中有一种不等半径的统计图称为南丁格尔玫瑰图,网上很热门,是一很有艺术感的漂亮的统计图,下面我们就来看看如何使用C语言绘制它吧
    2024-03-03
  • C++中Lambda表达式的语法与实例

    C++中Lambda表达式的语法与实例

    C++ 11 中的 Lambda 表达式用于定义并创建匿名的函数对象,以简化编程工作,下面这篇文章主要给大家介绍了关于C++中Lambda表达式的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2021-10-10
  • C++中auto类型说明符详解(附易错实例)

    C++中auto类型说明符详解(附易错实例)

    这篇文章主要给大家介绍了关于C++中auto类型说明符的相关资料,文中还附易错实例,在C++11中引入了auto类型说明符,用它就能让编译器替我们去分析表达式所属的类型,需要的朋友可以参考下
    2023-07-07
  • C语言实现顺序表的基本操作指南(注释很详细)

    C语言实现顺序表的基本操作指南(注释很详细)

    线性表是最简单的数据结构,而顺序表又是最简单的线性表,其基本思想是用一段地址连续的储存单元依次存储线性表的数据元素,下面这篇文章主要给大家介绍了关于C语言实现顺序表的基本操作,需要的朋友可以参考下
    2021-10-10
  • c语言字符串函数strstr,strtok,strerror的使用和实现

    c语言字符串函数strstr,strtok,strerror的使用和实现

    C语言中的字符串处理函数如strtok、strstr和strerror对于字符串的处理有着重要的作用,strtok函数用于分割字符串,它通过sep参数指定的分隔符来分割str参数指定的字符串,并返回分割后的每个子字符串
    2024-10-10
  • C++模板以及实现vector实例详解

    C++模板以及实现vector实例详解

    模板是为了实现泛型编程,所谓泛型编程,就是指编写与类型无关的代码,下面这篇文章主要给大家介绍了关于C++模板以及实现vector的相关资料,需要的朋友可以参考下
    2021-11-11
  • Opencv基于CamShift算法实现目标跟踪

    Opencv基于CamShift算法实现目标跟踪

    这篇文章主要为大家详细介绍了Opencv基于CamShift算法实现目标跟踪,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • Cocos2d-x UI开发之场景切换代码实例

    Cocos2d-x UI开发之场景切换代码实例

    这篇文章主要介绍了Cocos2d-x UI开发之场景切换代码实例,cocos2d-x中的场景切换是通过导演类调用相应的方法完成的,本文通过代码和详细注释来说明,需要的朋友可以参考下
    2014-09-09

最新评论