0

0

C++如何实现自动化的内存泄漏检测工具?(自研诊断方案)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-03-14 20:50:31

|

757人浏览过

|

来源于php中文网

原创

运行时拦截 malloc/new 调用并记录调用栈、大小、地址等信息,Linux 用 LD_PRELOAD 重定义内存函数,Windows 用 Detour 或重载 operator new,配合文件行号宏和元数据头,退出前安全 dump 泄漏摘要。

c++如何实现自动化的内存泄漏检测工具?(自研诊断方案)

malloc / new 拦截 + 全局钩子就能拿到分配现场

核心思路不是靠事后扫描内存,而是运行时拦截所有堆分配调用,记录调用栈、大小、地址。C++ 没有标准 API 做这个,但 Linux 下可改写 __libc_malloc__libc_free,Windows 下用 DetourAttach 或 MSVC 的 _malloc_hook(注意:后者在较新 CRT 中已弃用)。

实操建议:

立即学习C++免费学习笔记(深入)”;

  • Linux 优先用 LD_PRELOAD 注入共享库,重定义 malloc/free/realloc/calloc,内部调用 __libc_malloc 等真实函数,避免递归
  • Windows 下若用 MSVC,_malloc_hook 仅支持 debug 版 CRT,且不捕获 new——必须同时替换全局 operator newoperator delete
  • 记录调用栈别依赖 backtrace()(信号安全问题),改用 libunwind 或 Windows 的 CaptureStackBackTrace
  • 每个分配记录至少存:ptrsizefilelinestack_hash(避免重复打印同一泄漏点)

如何让 operator new 自动带文件/行号信息?

标准 operator new 不传位置信息,但 C++17 起支持 operator new(std::size_t, std::source_location),不过主流编译器还没完全落地。更稳的方案是宏替换 + 重载带参版本。

实操建议:

立即学习C++免费学习笔记(深入)”;

  • 定义宏:#define NEW new (__FILE__, __LINE__),再声明 void* operator new(std::size_t, const char*, int)
  • 该重载函数里调用 malloc,并把文件/行号存入分配头(比如在 ptr 前面放 16 字节元数据)
  • 注意:类内自定义 operator new 会屏蔽全局重载,所以必须确保所有类没声明自己的 new,或统一继承一个基类来接管
  • 别忘了配对实现 operator delete(void*, const char*, int),否则 delete 时无法定位元数据

程序退出时 dump 未释放块,但 atexit 太晚了

atexit 注册的函数在静态析构之后执行,此时部分全局对象(比如日志模块)可能已销毁,导致写文件失败或崩溃。真正安全的时机是 main 返回后、CRT 清理前。

Insou AI
Insou AI

Insou AI 是一款强大的人工智能助手,旨在帮助你轻松创建引人入胜的内容和令人印象深刻的演示。

下载

实操建议:

立即学习C++免费学习笔记(深入)”;

  • Linux 下用 __cxa_atexit 注册函数,并指定一个比默认优先级更高的序列值(如 -1),让它早于静态析构执行
  • Windows 下用 onexit(仅 MSVC)或直接 hook exit 函数,但需小心绕过 CRT 的双重检查
  • dump 内容别写磁盘——用 write(2) 直接写 stderr 或临时 fd,避免触发 std::ofstream 构造
  • 只打印泄漏块的 stack_hash 和总字节数,按哈希聚合,否则上万次分配会输出几 MB 无意义日志

为什么 valgrind 不够用,还得自己写?

valgrind 是重量级动态二进制插桩,启动慢、性能降 20–50 倍,且不支持某些内联汇编或硬件加速路径;更重要的是它看不到 C++ 对象语义(比如某 std::vector 的 capacity 是否被反复 realloc 却未 shrink_to_fit)。

实操建议:

立即学习C++免费学习笔记(深入)”;

  • 自研工具要轻量:单个共享库注入,运行时开销控制在 5% 以内(用 per-CPU 缓存分配记录,避免锁)
  • 加白名单机制:std::string 内部缓存、std::shared_ptr 控制块这类“合法长期持有”内存要过滤,否则误报爆炸
  • 支持运行时开关:setenv("LEAK_CHECK=0", 1, 1) 可在关键路径关闭检测,而不是重新编译
  • 最易忽略的一点:多线程下 pthread_atfork 必须注册 fork 后的清理逻辑,否则子进程会继承父进程的分配表并误报泄漏

真正难的不是记下每次 malloc,而是判断“什么时候算漏”。这取决于你的业务语义——比如网络模块里一个连接对象存活期间申请的 buffer,连接 close 后没释放才算漏。工具只能给原始数据,判定逻辑得你写。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1051

2023.08.02

typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

119

2023.09.26

define的用法
define的用法

define用法:1、定义常量;2、定义函数宏:3、定义条件编译;4、定义多行宏。更多关于define的用法的内容,大家可以阅读本专题下的文章。

391

2023.10.11

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

564

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1051

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

615

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

335

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11.3万人学习

C 教程
C 教程

共75课时 | 5.4万人学习

C++教程
C++教程

共115课时 | 21.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号