新闻中心

c++怎么实现一个简单的内存泄漏检测工具_c++自定义分配器与内存跟踪实现

2025-11-19
浏览次数:
返回列表
答案:通过重载new/delete并记录调用栈实现内存泄漏检测。1. 拦截new/delete操作,记录分配信息;2. 用静态析构函数输出未释放内存;3. 自定义分配器跟踪STL容器;4. 编译需-g和-rdynamic等选项支持符号解析。

c++怎么实现一个简单的内存泄漏检测工具_c++自定义分配器与内存跟踪实现

要实现一个简单的内存泄漏检测工具,核心思路是拦截所有内存分配和释放操作,记录每次分配的调用栈、大小和地址,并在程序结束时检查未释放的内存块。结合 C++ 自定义分配器与内存跟踪机制,可以做到轻量级但有效的泄漏监控。

1. 拦截 new/delete 操作

通过重载全局 operator newoperator delete,我们可以捕获所有动态内存操作:

#include <unordered_map>
#include <mutex>
#include <execinfo.h>
#include <cstdint>
<p>struct AllocInfo {
size_t size;
void* call_stack[10];
int stack_size;
};</p><p>std::unordered_map<void*, AllocInfo> g_allocations;
std::mutex g_alloc_mutex;</p><p>void<em> operator new(size_t size) {
void</em> ptr = std::malloc(size);</p><pre class="brush:php;toolbar:false;">if (!ptr) throw std::bad_alloc();

// 获取调用栈
AllocInfo info;
info.size = size;
info.stack_size = backtrace(info.call_stack, 10);

std::lock_guard<std::mutex> lock(g_alloc_mutex);
g_allocations[ptr] = info;

return ptr;

}

void operator delete(void* ptr) noexcept { if (!ptr) return;

std::lock_guard<std::mutex> lock(g_alloc_mutex);
g_allocations.erase(ptr);

std::free(ptr);

}

2. 添加静态析构函数输出泄漏报告

利用 C++ 静态对象在程序退出时自动析构的特性,在最后打印未释放的内存块:

struct LeakReporter {
    ~LeakReporter() {
        if (g_allocations.empty()) {
            printf("✅ No memory leaks detected.\n");
            return;
        }
<pre class="brush:php;toolbar:false;">    printf("❌ Memory leaks detected:\n");
    for (auto& pair : g_allocations) {
        printf("  Address %p, Size %zu bytes\n", pair.first, pair.second.size);
        char** symbols = backtrace_symbols(pair.second.call_stack, pair.second.stack_size);
        for (int i = 0; i < pair.second.stack_size; ++i) {
            printf("    %s\n", symbols[i]);
        }
        free(symbols);
    }
}

};

static LeakReporter reporter; // 全局静态实例

3. 使用自定义分配器进行容器内存跟踪

如果你希望对 STL 容器也做内存控制,可以实现一个自定义分配器:

小云雀 小云雀

剪映出品的AI视频和图片创作助手

小云雀 1949 查看详情 小云雀

template<typename T>
struct TrackingAllocator {
    using value_type = T;
<pre class="brush:php;toolbar:false;">T* allocate(size_t count) {
    void* ptr = ::operator new(count * sizeof(T));

    AllocInfo info{count * sizeof(T)};
    info.stack_size = backtrace(info.call_stack, 10);

    std::lock_guard<std::mutex> lock(g_alloc_mutex);
    g_allocations[ptr] = info;

    return static_cast<T*>(ptr);
}

void deallocate(T* ptr, size_t count) noexcept {
    std::lock_guard<std::mutex> lock(g_alloc_mutex);
    g_allocations.erase(ptr);
    ::operator delete(ptr);
}

};

然后用于标准容器:

std::vector<int, TrackingAllocator<int>> vec;
vec.push_back(42);  // 分配会被记录

4. 编译与使用注意事项

这个工具依赖于 backtrace(),仅在 Linux/macOS 的 GCC/Clang 下有效。编译时需加:

  • -g:保留调试信息
  • -fno-omit-frame-pointer:保证调用栈可追溯
  • -rdynamic(Linux)或 -export-dynamic:导出符号便于解析

示例编译命令:

g++ -g -fno-omit-frame-pointer -rdynamic main.cpp -o leak_test

基本上就这些。这个方案不复杂但容易忽略细节,比如线程安全和异常安全。实际项目中可进一步扩展支持文件名/行号(借助宏注入),或写入日志文件。对于生产环境,建议结合 Valgrind 或 ASan 使用,但自研工具在定制化和性能监控上仍有优势。

以上就是c++++怎么实现一个简单的内存泄漏检测工具_c++自定义分配器与内存跟踪实现的详细内容,更多请关注其它相关文章!


# 网络编程  # 营销策略推广方式  # 画家营销推广团队有哪些  # seo源码后台  # 七七seo  # 大连关键词排名优化打造  # 关键词点击排名技巧  # 深泽软文网站推广  # 杭州网站快速推广  # 评多多关键词搜索有排名  # 佛山新网站建设最专业  # 相关文章  # 我们可以  # 并在  # 如果你  # linux  # 有什么区别  # 第三方  # 行号  # 微软  # 自定义  # red  # cos  # macos  # c++  # ai  #   # mac  # 工具 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: Android Studio计算器C键功能异常排查与修复教程  Angular响应式表单:实现提交后表单及按钮的禁用与只读化  小猿搜题在线学习页面在哪_小猿搜题在线学习中心入口  Yandex浏览器官方网页版入口 Yandex浏览器最新版官网  EMS快递官网app_中国邮政速递物流手机客户端  QQ邮箱网页版入口页面 QQ邮箱在线登录入口官网  Linux如何排查内存不足OOME问题_LinuxOOM分析教程  优酷会员付费后没到账怎么办_优酷会员充值异常及解决方法  J*aScript对象创建方式_J*aScript设计模式应用  QQ邮箱官方邮箱登录入口 QQ邮箱网页版快速访问  Composer的 "check-platform-reqs" 命令有什么用_在部署前检查生产环境是否满足Composer依赖需求  必由学官方登录入口 必由学教师学生账号快速访问  Mac怎么使用表情符号_Mac Emoji快捷键面板  Lar*el递归关系中排除子孙节点的策略  在J*a项目里如何构建对象之间的契约_接口约束的实际落地  高德地图沿途添加点失败如何解决 高德多点规划方法  谷歌浏览器如何快速清除某个网站的数据_Chrome网站缓存清理方法  Golang如何使用net/url解析URL_Golang URL解析与处理方法  《明末:渊虚之羽》设计师谈设计角色:那会刚毕业 充满激情  CSS布局中意外空白:解决padding-top导致的顶部间距问题  菜鸟取件码是什么怎么查 最全查询渠道汇总  AO3官方在线访问地址 Archive of Our Own最新镜像合集  Go语言中高效处理x-www-form-urlencoded表单数据  Composer的 "conflict" 字段有什么用_如何声明不兼容的包以避免依赖冲突  新三国志曹操传110级星符试炼夏侯渊极难攻略  Golang如何实现Web文件静态资源服务器_Golang静态资源服务器开发与实践  期待已久:小米17 Ultra、小米首款NAS本月登场  J*aScript中在Map循环中检测并处理空数组元素  Word2013如何插入视频和音频媒体_Word2013媒体插入的多媒体支持  mc.js游戏直达 mc.js网页免下载版本秒进地址  Mac怎么锁定备忘录_Mac备忘录加密设置教程  C++如何生成随机数_C++ random库使用方法与范围设置  智慧团建扫码登录入口 智慧团建扫码登录入口官网版​  CSS如何设置hover状态颜色_hover伪类调整背景或文字颜色  AO3镜像入口大全 AO3网页版内容访问全集  Log4j Console Appender性能瓶颈与高并发优化策略  响应式图片在网页设计中的正确实现方法  windows10怎么查看硬盘序列号_windows10硬盘id查询命令  qq游戏网页版直接玩_qq游戏免下载快速入口  AO3最新官网入口公告_2025AO3镜像站实时查询方法  UE5.7引擎表现爆炸优化无敌!5090跑4K稳定60FPS  mysql密码锁定怎么解锁_mysql密码锁定解锁后修改密码步骤  VS Code远程开发时如何处理文件权限问题  Win10如何清理注册表垃圾 Win10注册表维护与优化指南【慎用】  css链接悬停下划线样式如何自定义_使用::after结合content和transition  特斯拉自动驾驶房车计划曝光 原型车将于2027年亮相  Composer的 "licenses" 命令如何帮助你遵守开源协议_检查项目依赖的许可证合规性  如何使用spryker/configurable-bundles-products-resource-relationship模块解决复杂产品捆绑关系难题  sublime怎么覆盖插件的默认快捷键_sublime快捷键优先级与设置  深入理解Go语言中的指针类型:以*string为例 

搜索