当前位置: 七九推 > 互联网>腾讯>微信 > HEAP: Free Heap block XXXX modified at XXXX after it was freed

HEAP: Free Heap block XXXX modified at XXXX after it was freed

2022年06月23日 微信 我要评论
*本文旨在解决在调试过程中遇到如下问题时的解决办法: HEAP: Free Heap block XXXXA modified at XXXXB after it was freed 意

*本文旨在解决在调试过程中遇到如下问题时的解决办法:

HEAP: Free Heap block XXXXA modified at XXXXB after it was freed

意思是:已经释放的内存地址A,在B地址处的值被改变(A和B都处于被释放的内存段内),即很可能出现了野指针,而很多情况下你会说,我的每个new和delete都是成对的,在delete后将指针赋值为NULL。但是我想说,野指针不单单会出现在变量中,它有可能会隐藏到你的类函数中,尤其是静态函数中。

先说说我遇到的问题。由于需要和下位机通过串口通信,就封装了一个串口操作的类CSerialPort作为基类,一个和下位机进行协议交互的类CSmartIOS2作为子类,父类在打开串口后会启动一个读写串口的线程,线程函数为基类的静态函数。在程序中调用此类进行通信,通信过程中没有任何问题,当应用程序结束时,我会关掉串口,释放掉CSmartIOS2的实例对象,问题来了,会报以上错误。一下子蒙逼了,在经历过把所有new和delete相关的内容都筛选后,仍然没找到可能出问题的地方。经过一番折腾,终于让我找到了一点规律,

HEAP: Free Heap block XXXXA modified at XXXXB after it was freed

XXXXA地址和XXXXB地址还有CSmartIOS2类实例的地址有如下关系:A地址始终比实例地址小,而且B地址和实例地址的差值始终等于64,这让我更加肯定了自己的猜想,改变的地址B是CSmartIOS2或其基类中的成员变量。

经过断点监测类实例地址,通过以上规律算出了类成员的地址,然后下内存断点,当该地址的值被改变时机会触发断点。如此终于找到了让此变量改变的地方,位于CSerialPort类的静态函数中,也是用于读写串口的线程,而这个地址就是类中的一个成员的地址。一切真像大白后,仍有一点不明,那就是明明我在delete 类实例前,先结束掉了串口读写线程,为何还会在delete后被该线程改变了内存呢?

我在关闭端口时,先通过

pBoard->ClosePort()

来结束掉IO线程,关闭端口。紧接着就做delete pBoard;动作,而判断线程结束的标识是位于线程函数返回前的一个布尔变量,当变量为真时,线程马上就结束了,但是是马上而不是已经结束,此时调用关闭端口的线程在判断IO线程变量为真后,立刻做delete pBoard,但是IO线程中的最后一句return 0有可能还没执行,当delete完成后,IO线程函数执行完毕,释放线程资源,此时又对类中的一个用于重叠结构的变量

OVERLAPPED m_ov;
的值被改变,问题就这样发生了,最简单的解决办法就是在结束掉线程后,适当等待一段时间给IO线程去释放资源。大概的代码片如下:

HANDLE yk_s2_InitBoard(UINT nPort)
{
	MyLock lock(list_lock);

	CSmartIOS2* pBoard = new CSmartIOS2();
	if(pBoard->Init(nPort))
	{
		int ipList[16] = {0};
		int nIP = 0;
		int result = pBoard->EnumIP(ipList,nIP);
		if(result == RET_NUMBER)
		{
			if(nIP > 0)
			{
				theBoardS2List.push_back(pBoard);
				return pBoard;
			}
		}
		else if(result == RET_INVALID_CMD)
		{
			theBoardS2List.push_back(pBoard);
			return pBoard;
		}
		pBoard->ClosePort();
	}
	delete pBoard;
	return NULL;
}


//关闭串口,释放内存
BOOL yk_s2_CloseBoard(HANDLE hBoard)
{
	if(hBoard == NULL)
		return FALSE;
	if(!isValidatePtr(hBoard))
		return FALSE;

	MyLock lock(list_lock);

	CSmartIOS2* pBoard = (CSmartIOS2*)hBoard;
	if(pBoard->IsOpen())
	{
		if(pBoard->ClosePort())
		{
			Sleep(500);//此处是整个问题的关键点
			std::list<CSmartIOS2*>::iterator iter;
			for(iter = theBoardS2List.begin();iter != theBoardS2List.end();)
			{
				if((*iter) == pBoard)
				{
					delete pBoard;
					iter = theBoardS2List.erase(iter);
					return TRUE;
				}
				else
					++iter;
			}
		}
		else
		{
			return FALSE;
		}
	}
	return TRUE;
}

总结如下:

遇到此问题后,首先应该大体定位出问题的内存地址大概是属于那个对象的,最好可以推出具体的某个变量。然后通过内存断点去找改变此地址值的代码段。然后定位问题,分析引起原因。

(0)
打赏 微信扫一扫 微信扫一扫

相关文章:

  • 微信开发者工具在哪查看缓存信息?微信开发者工具查看缓存信息教程

    微信开发者工具在哪查看缓存信息?微信开发者工具查看缓存信息教程

    微信开发者工具如何查看缓存信息呢?今天小编就为小伙伴们带来了微信开发者工具查看缓存信息的详细步骤,一起来看看吧。微信开发者工具查看缓存信息教程首先,打开微信开发... [阅读全文]
  • Codeforces Round #658 (Div. 2) B. Sequential Nim

    Codeforces Round #658 (Div. 2) B. Sequential Nim

    B. Sequential Nim题目链接-B. Sequential Nim题目大意给 nnn 堆石子的数量,两人轮流从最左端的非空堆中取任意数量(>0... [阅读全文]
  • 扫描全能王如何取消自动扣费 关闭自动续费方法

    图片扫描全能王 for android v1.6.7 安卓手机版类型:商务办公大小:41.6mb语言:简体中文时间:2021-01-06扫描全能王会员服务该怎么关闭扣费服务呢?有很…

    2022年06月23日 腾讯
  • obsidian怎么设置链接?obsidian设置链接教程

    想要在obsidian中插入链接该怎么操作呢?插入链接作为obsidian最大的特色,喜欢用obsidian这款软件的小伙伴可不能错过,下面就和小编一起来看看吧。obsidian链…

    2022年06月23日 腾讯
  • 喜马拉雅如何把声音加到听单?

    我们有时候在使用喜马拉雅的时候,想要把声音加到听单,如何操作呢,感兴趣的朋友不要错过了。方法1、打开“喜马拉雅"app。2、进入主界面后点击中间的&ldquo…

    2022年06月23日 腾讯
  • 微信背景图应用到全部场景的操作方法

    微信背景图怎么应用到全部场景的呢?下面是小编介绍微信背景图应用到全部场景的操作方法过程,有需要的小伙伴一起来下文看看吧,希望可以帮助到大家!微信背景图应用到全部场景的操作方法1、首…

    2022年06月23日 腾讯

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2022  七九推 保留所有权利. 粤ICP备17035492号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com