C++程序在Windows平台上各种定位内存泄漏的方法

一、前言

        在Linux平台上有valgrind可以非常方便的帮助我们定位内存泄漏,因为Linux在开发领域的使用场景大多是跑服务器,再加上它的开源属性,相对而言,处理问题容易形成“统一”的标准。而在Windows平台,服务器和客户端开发人员惯用的调试方法有很大不同。下面结合我的实际经验,整理下常见定位内存泄漏的方法。

注意:我们的分析前提是Release版本,因为在Debug环境下,通过VLD这个库或者CRT库本身的内存泄漏检测函数能够分析出内存泄漏,相对而言比较简单。而服务器有很多问题需要在线上并发压力情况下才出现,光棍影院因此讨论Debug版调试方法意义不大。

二、对象计数

方法:在对象构造时计数++,析构时--,每隔一段时间打印对象的数量

优点:没有性能开销,几乎不占用额外内存。定位结果精确。

缺点:侵入式方法,需修改现有代码,而且对于第三方库、STL容器、脚本泄漏等因无法修改代码而无法定位。

三、重载new和delete

方法:重载new/delete,记录分配点(甚至是调用堆栈),定期打印。

优点:没有看出

缺点:侵入式方法,需将头文件加入到大量源文件的头部,以确保重载的宏能够覆盖所有的new/delete。记录分配点需要加锁(如果你的程序是多线程),而且记录分配要占用大量内存(也是占用的程序内存)。

四、Hook Windows系统API

方法:使用微软的detours库,hook分配内存的系统Api:HeapAlloc/HeapRealloc/HeapFree(new/malloc的底层调用),记录分配点,定期打印。

优点:非侵入式方法,无需修改现有文件新视觉影院(hook api后,分配和释放走到自己的钩子函数中),检查全面,对第三方库、脚本库等等都能统计到。

缺点:记录内存需要占用大量内存,而且多线程环境需要加锁。

五、使用DiagLeak检测

微软出品的内存泄漏分析工具,原理同hookapi方式。配合LDGraph可视化展示内存分配数据,更方便查找泄漏。

在IDE工程选项里面配置Release版本也生成调试信息,发布时,将pdb文件和exe文件一起发布。

C++程序如何精确查找内存泄漏

程序运行后,打开LeakDiag,设置Symbol path

C++程序如何精确查找内存泄漏

定期Log下目标进程的内存分配情况,通过LDGraph打印分配增长情况,来发现内存泄漏。

C++程序如何精确查找内存泄漏

优点:同hookapi方法,非侵入式修改,无需做任何代码改动。跟踪全面。可视化分析堆栈一览无余!

缺点:对性能有影响,hook分配加锁,遍历堆栈。但是不会占用目标进程的自身内存。

六、总结

对于线上生产环境,建议大对象用计数来判断,定位快速准确,几乎无性能开销。在对外测试阶段,使用LeakDiag辅助分析,因为此时并发压力还不 是太大,性能开销还是可以承受。在线上大规模应用阶段,通过HookApi的方法,结合GM指令控制部分时间段的检测,这样可以把对玩家的影响(服务器性 能下降导致延迟)降到最低。

Windows的C编程中,定位内存错误是一项非常重要的任务,因为内存错误可能导致程序崩溃、数据损坏或安全漏洞。以下是一些常见的内存错误类型及其解决方法: 1. 内存泄漏:当程序分配了内存空间但未及时释放时,就会发生内存泄漏。这可能导致内存不足,造成程序崩溃。解决内存泄漏方法是在不再使用内存时手动释放它,使用free()函数或类似的内存释放函数。 2. 空指针引用:若使用未初始化或已释放的指针,程序就会发生空指针引用错误。解决这个问题的方法是确保指针被正确初始化,并且在使用指针之前检查其是否为空。可以使用条件判断来避免空指针引用。 3. 缓冲区溢出:当程序向一个已满的缓冲区写入数据时,就会发生缓冲区溢出错误。这可能导致覆盖其他变量或执行异常代码。为了解决这个问题,需要明确缓冲区的最大长度,并在写入数据之前检查缓冲区的可用空间。 4. 访问越界:当程序试图访问数组或其他数据结构中不存在的元素时,就会发生访问越界错误。这可能导致数据损坏或崩溃。要解决这个问题,应该确保在访问数组元素之前检查索引是否在有效范围内。 5. 释放已释放的内存:在程序中重复释放已经释放的内存可能导致内存错误。为了避免这个问题,应该跟踪每块内存的分配和释放情况,并确保每次释放都是针对已分配内存的合法操作。 总之,找出并解决Windows C程序中的内存错误对于确保程序的稳定性和安全性至关重要。通过正确地分配和释放内存、检查指针和索引的合法性,可以有效防止内存错误的发生。同时,使用工具如静态代码分析器和内存检测工具也能帮助发现并解决内存错误。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值