智能指针-share_ptr循环引用导致内存泄漏

2024-06-16 23:48

本文主要是介绍智能指针-share_ptr循环引用导致内存泄漏,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

有两个栈内share_ptr对象aptr/bptr分别指向两个对象A/B,A/B两个对象互相使用一个shared_ptr成员变量指向对方A/B,会造成循环引用。

当两个栈内存离开main函数作用域后只是对象对象A/B引用计数减少1,但A/B对象的引用计数并没有减少到0,因此内存不会被share_ptr的析构函数给释放,造成内存泄漏


#include<iostream>
#include<memory>
using namespace std;class B;class A {
public:~A();void setS(const shared_ptr<B> &bp);
private:shared_ptr<B> bPtr;
};void A::setS(const shared_ptr<B> &bp)
{bPtr = bp;
}A::~A()
{cout << "~A" << endl;
}class B {
public:~B();void setS(const shared_ptr<A> &ap);
private:shared_ptr<A> aPtr;
};void B::setS(const shared_ptr<A> &ap)
{aPtr = ap;
}B::~B()
{cout << "~B" << endl;
}int main()
{cout << "Hello world!" << endl;shared_ptr<A> aptr = make_shared<A>();shared_ptr<B> bptr = make_shared<B>();int aRefCnt = aptr.use_count();int bRefCnt = bptr.use_count();int aRefCntAfter = -1;int bRefCntAfter = -1;if (aptr != nullptr && bptr != nullptr) {aptr->setS(bptr);bRefCntAfter = bptr.use_count(); // 2bptr->setS(aptr);aRefCntAfter = aptr.use_count(); // 2}// 所以申请的对象现在引用计数都是2,,main函数结束后,// 虽局部变量aptr和bptr会栈回收,但指向的对象应用计数仍为1,永远不会调用析构来释放内存。return 0;
}

 

参考:

https://www.cnblogs.com/wxquare/p/4759020.html

https://www.cnblogs.com/ymd12103410/p/11136584.html

这篇关于智能指针-share_ptr循环引用导致内存泄漏的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1067893

相关文章

golang内存对齐的项目实践

《golang内存对齐的项目实践》本文主要介绍了golang内存对齐的项目实践,内存对齐不仅有助于提高内存访问效率,还确保了与硬件接口的兼容性,是Go语言编程中不可忽视的重要优化手段,下面就来介绍一下... 目录一、结构体中的字段顺序与内存对齐二、内存对齐的原理与规则三、调整结构体字段顺序优化内存对齐四、内

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

Linux内存泄露的原因排查和解决方案(内存管理方法)

《Linux内存泄露的原因排查和解决方案(内存管理方法)》文章主要介绍了运维团队在Linux处理LB服务内存暴涨、内存报警问题的过程,从发现问题、排查原因到制定解决方案,并从中学习了Linux内存管理... 目录一、问题二、排查过程三、解决方案四、内存管理方法1)linux内存寻址2)Linux分页机制3)

Python判断for循环最后一次的6种方法

《Python判断for循环最后一次的6种方法》在Python中,通常我们不会直接判断for循环是否正在执行最后一次迭代,因为Python的for循环是基于可迭代对象的,它不知道也不关心迭代的内部状态... 目录1.使用enuhttp://www.chinasem.cnmerate()和len()来判断for

Java循环创建对象内存溢出的解决方法

《Java循环创建对象内存溢出的解决方法》在Java中,如果在循环中不当地创建大量对象而不及时释放内存,很容易导致内存溢出(OutOfMemoryError),所以本文给大家介绍了Java循环创建对象... 目录问题1. 解决方案2. 示例代码2.1 原始版本(可能导致内存溢出)2.2 修改后的版本问题在

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

SpringBoot项目中Maven剔除无用Jar引用的最佳实践

《SpringBoot项目中Maven剔除无用Jar引用的最佳实践》在SpringBoot项目开发中,Maven是最常用的构建工具之一,通过Maven,我们可以轻松地管理项目所需的依赖,而,... 目录1、引言2、Maven 依赖管理的基础概念2.1 什么是 Maven 依赖2.2 Maven 的依赖传递机

Redis多种内存淘汰策略及配置技巧分享

《Redis多种内存淘汰策略及配置技巧分享》本文介绍了Redis内存满时的淘汰机制,包括内存淘汰机制的概念,Redis提供的8种淘汰策略(如noeviction、volatile-lru等)及其适用场... 目录前言一、什么是 Redis 的内存淘汰机制?二、Redis 内存淘汰策略1. pythonnoe

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J