内容
介绍
内存泄漏经常发生在 C/C++ 代码中,尤其是在具有大量遗留代码库的大型项目中。如果一个庞大的开发团队维护一个项目,那么这种泄漏可能会“存在”多年,而没有机会见面并修复它。
项目中存在与内存相关的错误的原因有多种。它们各不相同,但主要是开发人员粗心、缺乏本机应用程序开发的技能或经验。一些从托管世界、Java、.NET 或 JavaScript 转向 C++ 的开发人员认为,删除操作符并不是强烈需要的,垃圾收集无论如何都能完成这项工作。嗯,听起来不错,但 C++ 没有。
无论开发人员多么有经验,内存泄漏仍然会发生。查找内存泄漏的过程通常是令人头脑麻木的。市场上有多种工具可以帮助调查泄漏;其中一些是免费的。本文将展示如何使用 WinDbg 应用程序修复 Windows 上的内存泄漏。此外,您还将了解如何使用替代工具 Deleaker,这是一个用于 C/C++ 的内存分析器。
WinDbg 是 Windows 调试工具的一部分。它是 Microsoft 提供的适用于内核和用户空间的强大调试器,也是查找内存泄漏的绝佳工具。WinDbg 可以在最复杂的情况下指出代码块,这些代码块可能是程序中内存泄漏的罪魁祸首。
Deleaker 也是 Windows 的内存泄漏检测工具。它可以作为独立工具或几乎所有流行 IDE 中的插件使用:Visual Studio、Qt Creator 和 RAD Studio。
Visual Studio 调试器和 C++ 运行时内存诊断工具通常可以提供有关内存泄漏来源的信息。然而,它们在使用中有点尴尬,因为它们需要访问源代码,有时需要调试版本。本文不重点介绍这种情况。当泄漏来自源代码不可用的外部模块时,这些工具就不起作用。在这种情况下,我们可以依靠WinDbg或Deleaker!
想象一个庞大而复杂的应用程序,它是由数十万行代码构建的。此外,您无权访问源代码。定位内存泄漏根源的过程听起来是一项具有挑战性的任务。但不用担心,因为至少有两种方法,读完本文后您就会明白。
您可以从这里下载 WinDbg 。Deleaker 可在官方网站上找到。
将两者安装在您的计算机上,我们就可以开始了。
内存泄漏的例子
常见的一个
让我们从一个简单且最常见的示例开始。在这种情况下,使用 C++ 中的运算符 new(或使用C 中的malloc函数)分配内存。尽管如此,它仍然没有对操作符delete(或C 中的free)进行适当的调用。该代码将分配的内存块的指针保存到变量中。正确的方法是使用这个变量来读写数据,一旦不需要该内存块,就使用操作符delete(或C中的free)来释放它。如果变量在释放内存之前获取另一个值,则没有机会释放内存:
1
2
3
4
5
6
7
8
|
void main()
{
auto p = malloc (32768);
memset (p, 0, 32768)
//free(p)
p = nullptr ;
}
|
这是另一种情况,动态分配的内存在函数的所有控制路径中都没有释放。例如,该函数调用TestFunc();反过来,TestFunc分配一个内存块并应在返回之前释放它。但是,失败时函数应返回一个或多个条件。因此,在某些情况下,代码不会释放分配的内存:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
void TestFunc( int value)
{
auto p = new int [32768];
if (value == 42)
|