`

C++内存泄露检查(Release)的5个方法

 
阅读更多
转自
http://www.open-open.com/lib/view/open1425869017857.html

一、前言
    在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可视化展示内存分配数据,更方便查找泄漏。

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


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


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


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

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

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

相关推荐

    检查内存泄露 检查内存泄露

    首先,检查内存泄露的一个常见方法是在程序的关键部分使用`CMemoryState`类,这是MFC库提供的一个工具,用于跟踪内存分配和释放的状态。`CMemoryState`提供了几个关键成员函数,包括: 1. **Checkpoint()**:这个...

    Qt内存溢出检测

    Visual Leak Detector是一个开源的C++内存泄漏检测器,专为Windows平台设计,可以与Visual Studio无缝集成。它通过在运行时捕获堆分配和释放的信息,找出未能释放的内存块。VLD的使用非常简单,只需要将其包含在项目...

    使用 CRT 调试功能来检测内存泄漏

    通过开启调试标志、使用调试版内存分配函数以及适时进行内存泄漏检查,开发者可以及时发现并修复内存泄漏问题,从而提高程序的稳定性和性能。在实际开发中,结合代码审查和自动化测试,可以进一步提升软件质量。

    C++MFC内存监视器.zip

    2. **内存泄漏检测**:通过跟踪分配和释放的内存块,检查是否有未释放的内存,这是内存监视器的核心功能之一。 3. **对象计数**:MFC中的`CObject`类提供了引用计数机制,通过`AddRef`和`Release`方法,可以跟踪...

    动态内存分配实验报告

    实验目的在于设计一个能够模拟内存分配状态的程序,通过实际操作理解内存分配的步骤和方法。实验要求包括以下几个方面: 1. 初始化内存:在程序开始时,需要设置一个初始的内存池,模拟系统的可用内存。在这个例子...

    [微软工具][内存溢出] LeakDiag & LDGrapher & LeakDiagAutoLog - 适用于 release后期压力测试

    5. 解决内存泄漏问题的方法和最佳实践。 6. 实际案例分析,展示如何利用这两个工具解决特定的内存泄漏问题。 了解并熟练使用这些工具对于任何涉及C++或其他类似内存管理语言的开发者来说都至关重要,因为它们能有效...

    vld(Visual Leak Detector 内存泄露检测工具 源码)

     Visual Leak Detector是一款用于Visual C++的免费的内存泄露检测工具。相比较其它的内存泄露检测工具,它在检测到内存泄漏的同时,还具有如下特点:  1、 可以得到内存泄漏点的调用堆栈,如果可以的话,还可以...

    c++ windows下限制软件只能打开一个(进程单开)

    在Windows环境下,开发C++应用程序时,有时我们需要确保同一时间只有一个实例运行,即实现进程的单开特性。这种功能在很多应用中都很常见,比如微信、QQ等,它避免了用户误操作导致多个实例同时运行,占用过多系统...

    Debug 和 Release 编译方式的本质区别

    - **Debug**:使用Debug版本的运行时刻库,该库通常包含额外的错误检测代码,例如内存泄漏检测、非法访问检测等。 - **Release**:使用优化后的运行时刻库,不包含额外的错误检测代码,以减小程序体积并提高运行...

    VS2008 Debug与Release的本质区别

    - `/GZ`:启用额外的内存检查,有助于发现内存泄漏等问题。 - `/Gm`:开启最小化重链接选项,以减少链接过程所需的时间。 - **Release版本**: - `/GF`:合并重复的字符串,并将字符串常量放入只读内存中,防止...

    C++调用vb6.0制作的Activex DLL

    5. **调用DLL方法**:通过接口指针调用ActiveX DLL的方法。注意,由于VB6.0的函数参数传递方式与C++不同,可能需要进行适当的类型转换。 例如,假设VB6.0的ActiveX DLL有一个名为`MyFunction`的函数,接受一个整数...

    Eliminating Memory Leaks in Symbian OS C++ Projects

    ### 消除Symbian OS C++项目中的内存泄漏 #### 概述 在Symbian OS中,内存管理是一项至关重要的任务,特别是在C++项目中。本文将深入探讨如何在Symbian OS环境中检测、避免内存泄漏,并提供具体方法来解决这些问题...

    c++dialog工程调用Com组件工程

    - 注意,不正确的引用计数管理可能导致内存泄漏或意外的组件销毁,因此在使用完组件后,必须调用`Release`。 5. **线程安全** - 如果你的对话框工程是多线程的,那么需要注意COM组件的线程模型。确保在正确的线程...

    Visual C++ 完全自学宝典 (部分章节) 电子书 PDF

    - **C#与Java**:提供垃圾回收机制,自动管理内存,减少了内存泄漏的风险。 - **是否跨平台** - **C++**:虽然标准C++本身是跨平台的,但由于它允许直接访问硬件和底层系统调用,因此实际的C++程序可能不是完全跨...

    示例代码:Release版崩溃,用VS调试dump文件,定位代码出错行【VS2017】

    例如,未捕获的异常、空指针引用、内存泄漏或访问越界都可能导致程序崩溃。在Release模式下,这些错误可能由于优化而变得更加难以发现,因此开发者需要对代码进行仔细的审查和测试。 总结来说,本示例代码的重点...

    Visual C++ 6.0 教程.zip

    5. 静态分析和性能优化:内置的代码分析工具可以帮助识别潜在的性能瓶颈和内存泄漏问题。 三、MFC 库详解: MFC 库是Visual C++ 6.0 中的核心部分,它封装了Windows API,提供了许多预先定义的类,如CWinApp、CWnd...

    release调试

    5. **调试工具的使用**:Visual Studio 自带了丰富的调试工具,如内存泄漏检测器、性能分析工具等,这些工具在 Release 构建中也同样有效。 6. **利用断点与跟踪点**:尽管优化可能导致某些变量位置发生变化,但...

    Visual_C++开发经验

    在Visual C++中,可以通过以下步骤在工作区中导入多个项目: - 首先,打开一个项目文件(通常是以`.dsp`为扩展名的文件)。 - 接下来,在菜单栏中选择 `Project` > `Insert Project into Workspace` 子菜单,选择另...

Global site tag (gtag.js) - Google Analytics