IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> shared_ptr循环引用问题以及解决方法 -> 正文阅读

[C++知识库]shared_ptr循环引用问题以及解决方法

一、shared_ptr循环引用问题

什么是循环引用,两个对象相互使用shared_ptr指向对方。造成的后果是:内存泄漏

例子一

下面是循环引用的例子

#include <iostream>
#include <memory>
using namespace std;


class A;
class B;

class A {
public:
    std::shared_ptr<B> bptr;
    ~A() {
        cout << "A is deleted" << endl; // 析构函数后,才去释放成员变量
    }
};

class B {
public:
    std::shared_ptr<A> aptr;
    ~B() {
        cout << "B is deleted" << endl;  // 析构函数后,才去释放成员变量
    }
};

int main()
{
    std::shared_ptr<A> pa;

    {
        std::shared_ptr<A> ap(new A);
        std::shared_ptr<B> bp(new B);
        ap->bptr = bp;
        bp->aptr = ap;
    }
    return 0;
}

这种状态下,它们的引用计数为均为2

{
	   std::shared_ptr<A> ap(new A);
	   std::shared_ptr<B> bp(new B);
	   ap->bptr = bp;
	   bp->aptr = ap;
}

在作用域内ap和bp的引用计数都为2,但是当它们退出循环的时候,ap的引用计数减1,bp的引用计数也减1,但它们依旧不为0,引用计数均为1。
对ap来说:只有调用了A的析构函数,才会去释放它的成员变量bptr。何时会调用A的析构函数呢?就是ap的引用计数为0
对于bp来说,只有调用了B的析构函数,才会去释放它的成员变量aptr。同样是bp的引用计数都为0的时候才能析构。

现在,对于ap和bp来说,它们都拿着对方的share_ptr(有点类似于死锁的现象),没法使得ab和bp的引用计数为0。那么A和B的对象均无法析构。于是造成了内存泄漏。

ap和bp退出作用域了,为什么不会调用析构函数呢?
ap和bp是创建在栈上的,而不是A或者B对象的本身,ap、bp退出作用域,只是ap和bp本身释放了,只会使得,A、B对象的引用计数-1,调用析构函数,是要A或B的对象,的引用计数为0才能执行析构函数。

例子二

如果将例子一,改成了下面这样,A对象的引用计数为1,B对象的引用计数为2
当ap和bp退出作用域时,
首先栈上的bp会被释放,那么B对象的引用计数-1,从2变为 1
然后栈上的ap会释放,那么A对象的引用计数-1,变成0。那么会调用A对象的析构函数,那么A对象中的成员bptr也会被释放,那么B对象的引用计数-1,也变成0,就会调用B对象的析构函数。

{
	   std::shared_ptr<A> ap(new A);
	   std::shared_ptr<B> bp(new B);
	   ap->bptr = bp;
	   //bp->aptr = ap;
}

执行结果
在这里插入图片描述
(注意,顺序是反着的,先释放栈上的bp,然后再是ap)

例子三

由于ap和bp都拿着对方的shared_ptr,导致循环引用。那么可以手动释放成员变量,比如将ap->bptr释放,那么此时B对象的引用计数为1,A对象的引用计数为2。

就是跟例子二类似的情况了,A对象和B对象都能够成功析构,不会造成内存泄漏。

{
        std::shared_ptr<A> ap(new A);
        std::shared_ptr<B> bp(new B);
        ap->bptr = bp;
        bp->aptr = ap;

        ap->bptr.reset(); // 手动释放成员变量才行
    }

二、weak_ptr解决循环引用问题

shared_ptr采用引用计数的方式,为0的时候就会去析构对象。
可以发现weak_ptr,不影响引用计数,是一种不控制对象生命周期的智能指针。

int main()
{
    shared_ptr<int> sp(new int(10));
    cout<<sp.use_count()<<endl; //输出1
    weak_ptr<int> wp1=sp;
    weak_ptr<int> wp2=sp;
    cout<<sp.use_count()<<endl; //输出1
}

因此只要将例子一中,类成员从shared_ptr改为weak_ptr,即可解决循环引用问题

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-06-06 17:10:30  更:2022-06-06 17:11:14 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 5:56:26-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码