C语言 智能指针 shared_ptr 和 weak_ptr

 更新时间:2022年04月11日 18:50:33   作者:哈利马其  
这篇文章主要介绍了C语言 智能指针 shared_ptr 和 weak_ptr,weak_ptr引入可以解决shared_ptr交叉引用时无法释放资源的问题,下面来学习具体相关内容吧,需要的朋友可以参考一下

weak_ptr引入可以解决shared_ptr交叉引用时无法释放资源的问题。

示例代码:

#include <iostream>
#include <memory>

using namespace std;

class B;

class A{
public:
    A(){cout << "A constructor ... "<< endl;}
    ~A(){cout << "A destructor ..." << endl;}
    std::shared_ptr<B> pb;
};

class B{
public:
    B(){cout << "B constructor ... "<< endl;}
    ~B(){cout << "B destructor ..." << endl;}
    std::shared_ptr<A> pa;
};

int main(int argc, char **argv) {
    
    std::shared_ptr<int> a = std::make_shared<int>(3);
    std::shared_ptr<char> b = std::make_shared<char>('a');
    
    std::cout << "shared_ptr object(int) size = " << sizeof(a) << std::endl;
    std::cout << "shared_ptr object(char) size = " << sizeof(b) << std::endl;
    
    std::weak_ptr<A> shadow_a;
    std::weak_ptr<B> shadow_b;
    
    {
    std::shared_ptr<A> ptr_a = std::make_shared<A>();
    std::shared_ptr<B> ptr_b = std::make_shared<B>();
    
    shadow_a = ptr_a;
    shadow_b = ptr_b;
    
    ptr_a->pb = ptr_b;
    ptr_b->pa = ptr_a;
    
    cout << "reference count of A = " << shadow_a.use_count() << endl;
    cout << "reference count of B = " << shadow_b.use_count() << endl;
    cout << endl; 
    }
    
    cout << "reference count of A = " << shadow_a.use_count() << endl;
    cout << "reference count of B = " << shadow_b.use_count() << endl;
    
    std::cout << "Hello, world!" << std::endl;
    return 0;
}

运行代码得到以下输出:

shared_ptr object(int) size = 16
shared_ptr object(char) size = 16
A constructor ... 
B constructor ... 
reference count of A = 2
reference count of B = 2

reference count of A = 1
reference count of B = 1
Hello, world!

从结果可以看出,由于交叉引用导致申请的内存A,B无法正常释放。
为什么会这样呢?这个应该从析构原理进行考虑,shared_ptr引用计数需要为0才会进行析构!但是ptr_a离开作用域会导致A引用计数减少1,但是A的引用计数此时为1,那么 pb不会释放;同理,ptr_b离开作用域会导致B引用计数减少1,但是B的引用计数为此时为1,那么pa不会释放。如此导致了资源无法释放掉。
由于weak_ptr并不会改变shared_ptr的引用计数,所以修改类A,和类B中的shared_ptr对象为weak_ptr对象即可释放资源。

修改后的代码如下:

#include <iostream>
#include <memory>

using namespace std;

class B;

class A{
public:
    A(){cout << "A constructor ... "<< endl;}
    ~A(){cout << "A destructor ..." << endl;}
    //std::shared_ptr<B> pb;
    std::weak_ptr<B> pb;
};

class B{
public:
    B(){cout << "B constructor ... "<< endl;}
    ~B(){cout << "B destructor ..." << endl;}
    //std::shared_ptr<A> pa;
    std::weak_ptr<A> pa;
};

int main(int argc, char **argv) {
    
    std::shared_ptr<int> a = std::make_shared<int>(3);
    std::shared_ptr<char> b = std::make_shared<char>('a');
    
    std::cout << "shared_ptr object(int) size = " << sizeof(a) << std::endl;
    std::cout << "shared_ptr object(char) size = " << sizeof(b) << std::endl;
    
    std::weak_ptr<A> shadow_a;
    std::weak_ptr<B> shadow_b;
    
    {
    std::shared_ptr<A> ptr_a = std::make_shared<A>();
    std::shared_ptr<B> ptr_b = std::make_shared<B>();
    
    shadow_a = ptr_a;
    shadow_b = ptr_b;
    
    ptr_a->pb = ptr_b;
    ptr_b->pa = ptr_a;
    
    cout << "reference count of A = " << shadow_a.use_count() << endl;
    cout << "reference count of B = " << shadow_b.use_count() << endl;
    cout << endl; 
    }
    
    cout << "reference count of A = " << shadow_a.use_count() << endl;
    cout << "reference count of B = " << shadow_b.use_count() << endl;
    
    std::cout << "Hello, world!" << std::endl;
    return 0;
}

运行结果如下,可以正常释放资源。

shared_ptr object(int) size = 16
shared_ptr object(char) size = 16
A constructor ... 
B constructor ... 
reference count of A = 1
reference count of B = 1

B destructor ...
A destructor ...
reference count of A = 0
reference count of B = 0
Hello, world!

到此这篇关于C语言 智能指针 shared_ptr 和 weak_ptr的文章就介绍到这了,更多相关 shared_ptr 和 weak_ptr内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++深入浅出讲解内存四区与new关键字的使用

    C++深入浅出讲解内存四区与new关键字的使用

    内存四区,一个非常重要的知识点,搞懂了内存四区,才能更快的去搞懂指针。我们写的C语言代码,不夸张的说,都是直接或者间接的在操作内存。C语言之所以能够开发操作系统,就是指针的存在,而指针说白了就是地址,内存地址,指针变量说白了就是存储地址的变量
    2022-05-05
  • Qt编写地图综合应用之绘制雨量分布

    Qt编写地图综合应用之绘制雨量分布

    雨量分布图是在区域地图基础上,针对区域中的每个最小单位区域比如县城点位不同颜色显示。本文将详细为大家介绍如何通过QT编写绘制雨量分布,感兴趣的小伙伴可以了解一下
    2021-12-12
  • Qt图形图像开发之曲线图表模块QChart库一个chart中显示两条曲线详细方法与实例

    Qt图形图像开发之曲线图表模块QChart库一个chart中显示两条曲线详细方法与实例

    这篇文章主要介绍了Qt图形图像开发之曲线图表模块QChart库一个chart中显示两条曲线详细方法与实例,需要的朋友可以参考下
    2020-03-03
  • C语言字符函数、内存函数功能及实现代码

    C语言字符函数、内存函数功能及实现代码

    这篇文章主要介绍了C语言字符函数、内存函数 功能及实现,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-02-02
  • C语言设计图书登记系统与停车场管理系统的实例分享

    C语言设计图书登记系统与停车场管理系统的实例分享

    这篇文章主要介绍了C语言设计图书登记系统与停车场管理系统的实例分享,重在以最简单的一些需求来展示管理系统的设计思路,需要的朋友可以参考下
    2016-06-06
  • 基于C++11实现手写线程池的示例代码

    基于C++11实现手写线程池的示例代码

    在实际的项目中,使用线程池是非常广泛的,本文主要介绍了基于C++11实现手写线程池的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-08-08
  • 基于C语言代码实现扫雷游戏

    基于C语言代码实现扫雷游戏

    这篇文章主要为大家详细介绍了基于C语言代码实现扫雷游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-11-11
  • c++ 入门——浅析构造函数和析构函数

    c++ 入门——浅析构造函数和析构函数

    这篇文章主要介绍了c++ 浅析构造函数和析构函数的相关资料,帮助大家入门c++ 编程,感兴趣的朋友可以了解下
    2020-08-08
  • C语言如何计算字符串长度

    C语言如何计算字符串长度

    这篇文章主要介绍了C语言如何计算字符串长度问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • C++实现中缀表达式转后缀表达式

    C++实现中缀表达式转后缀表达式

    这篇文章主要为大家详细介绍了C++实现中缀表达式转后缀表达式,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-04-04

最新评论