我在使用 SDL 库时也遇到了类似的问题。但就我而言,我试图使用 Visual Studio 2010 的默认内存泄漏检测,因为我不想使用第三方库/应用程序。
解决问题
如果在所有必需的包含、定义和函数调用之后您仍然看不到任何内存泄漏打印出来,则可能是您的运行时库设置不正确。
仔细检查您是否拥有运行时库的调试版本而不是非调试版本(/MT 和 /MD)。
多线程调试 (/MTd)
多线程调试 DLL (/MDd)
当您指定 /MTd 或 /Mdd 选项时,编译器会定义 _DEBUG。这些选项指定 C 运行时库的调试版本。见_DEBUG参考MSDN
因此,必须定义 _DEBUG 符号才能启用 CRT 代码。
[...] 如果未定义 _DEBUG,则在预处理过程中删除对 _CrtSetDbgFlag 的调用 [...]。请参阅 MSDN 参考
因此,构建调试版本不足以确保定义 _DEBUG。
这是您在普通项目中通常不会更改的内容,但遵循 SDL 教程可能会引导您成为我曾经的样子。
希望它会帮助别人,甚至是你。
更多详情如下
我正在关注MSDN 页面以启用 VS 2010 开箱即用的内存泄漏检测。
在声明那些
#define _CRTDBG_MAP_ALLOC
#include <stdlib.h>
#include <crtdbg.h>
我在我的代码中启用了它们,并故意插入了内存泄漏
int main( int argc, char* args[] )
{
_CrtSetDbgFlag( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
_CrtSetReportMode( _CRT_ERROR, _CRTDBG_MODE_DEBUG );
int *pArray = (int*)malloc(sizeof(int) * 24); // Memory not freed
return 0;
}
什么都没有打印出来。
所以,我查看了程序集,它肯定没有生成 CRT 代码,如您所见:
int main( int argc, char* args[] )
{
012932F0 push ebp
012932F1 mov ebp,esp
012932F3 sub esp,0CCh
012932F9 push ebx
012932FA push esi
012932FB push edi
012932FC lea edi,[ebp-0CCh]
01293302 mov ecx,33h
01293307 mov eax,0CCCCCCCCh
0129330C rep stos dword ptr es:[edi]
_CrtSetDbgFlag( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
_CrtSetReportMode( _CRT_ERROR, _CRTDBG_MODE_DEBUG ); // Nothing in both case!
int *pArray = (int*)malloc(sizeof(int) * 24);
0129330E mov esi,esp
01293310 push 60h
01293312 call dword ptr [__imp__malloc (129E4CCh)]
01293318 add esp,4
0129331B cmp esi,esp
0129331D call @ILT+580(__RTC_CheckEsp) (1291249h)
01293322 mov dword ptr [pArray],eax
然后,我意识到_DEBUG 符号可能没有被定义。