




Windows下用CRT库检测内存泄漏最直接,Linux下必须用Valgrind;两者原理不同,不能混用,也不能只依赖一个。
Windows 下用 CRT 库检测内存泄漏最直接,Linux 下必须用 Valgrind;两者原理不同,不能混用,也不能只依赖一个。
_CrtSetDbgFlag 和 _CrtDumpMemoryLeaks 检测泄漏CRT(C Runtime)库自带轻量级泄漏检测,仅限 Debug 模式 + MSVC 编译器,运行时自动记录堆分配/释放,程序退出前比对未释放块。
main() 开头调用 _CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF),否则不生效new 但没配 #include 或没定义 _CRTDBG_MAP_ALLOC,泄漏行号会显示为 malloc.c 而非你的源文件VirtualAlloc 等系统级分配,只管 malloc/new 经 CRT 分配的堆内存valgrind --leak-
check=full
Valgrind 是动态二进制插桩工具,不依赖编译器或运行时库,能覆盖所有堆分配(malloc、new、realloc),还能定位未初始化内存读写和越界访问。
-g,否则泄漏报告只有地址无行号;建议同时加 -O0 避免内联干扰调用栈valgrind --leak-check=full --show-leak-kinds=all ./myapp;--show-leak-kinds=all 才能看见 “still reachable” 类泄漏(常被误认为安全,实际可能是资源未释放)--trace-children=yes 并确认子进程也带调试信息)perf_event_open),遇到 crash 可尝试加 --tool=memcheck --track-origins=no 减负这两者都只监控用户态堆分配,以下情况完全静默:
std::string、std::vector 等容器内部扩容导致的临时内存未释放(实际是逻辑错误,不是堆泄漏)mmap(MAP_ANONYMOUS) 或 VirtualAlloc 分配的大块内存,绕过 malloc 堆管理器new 重载、OpenCV 的 cv::Mat::data)未按预期释放,而你没覆写其析构逻辑std::shared_ptr 互相持有),CRT/Valgrind 显示“已释放”,但对象实际未 destruct真正难查的是跨模块、延迟释放、或与 RAII 对象生命周期错位的泄漏——这时候得结合 addr2line、堆栈采样(perf record -e mem:swapped:u)或自定义 new/delete 拦截器,而不是只盯着 CRT 或 Valgrind 的输出行数。