0

0

C++如何利用Valgrind排查程序的非法内存访问?(报错解决)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-03-14 18:39:32

|

966人浏览过

|

来源于php中文网

原创

Valgrind报Invalid read/write时,首要排查栈溢出或越界,常见原因包括数组下标越界、使用已释放堆内存、局部变量地址逃逸;需启用--track-origins=yes并加-g、-O0编译。

c++如何利用valgrind排查程序的非法内存访问?(报错解决)

Valgrind报Invalid read/write时,先确认是不是栈溢出或越界

这类错误最常见的是数组下标越界、使用了已释放的堆内存,或者局部变量地址逃逸。Valgrind默认只检测堆内存(malloc/new),但Invalid read of size X也可能是栈上越界——比如char buf[10]; strcpy(buf, big_str);。这时候要加--tool=memcheck --track-origins=yes,让Valgrind追踪未初始化值的来源,否则容易误判为“野指针”。

  • 编译时务必加-g,否则行号全丢,定位困难
  • 禁用编译器优化(-O0),-O2可能把变量优化进寄存器,Valgrind看不到访问路径
  • 如果程序用了mmap或自定义分配器,需配合--freelist-vol=... --freelist-big-blocks=...调大缓存,否则漏报

遇到Address 0x... is 8 bytes inside a block of size 16 alloc'd说明真越界了

这条信息其实已经告诉你:访问地址在某次malloc返回块的内部偏移8字节处,而该块总共才16字节——也就是说,你读/写了第9~16字节之后的位置(比如索引[16][20])。关键不是看“inside”,而是对比“size”和你代码里实际访问的偏移量。

  • 检查所有for (int i = 0; i 这种 <li>注意<code>std::vector::data().size()的配合,别拿vec.data()[vec.size()]当合法地址
  • 多线程场景下,这个错误可能掩盖真正的竞态:一个线程刚delete,另一个线程立刻读,Valgrind会报Invalid read而非use-after-free(后者需要--tool=helgrind

Use of uninitialised value of size 8常被当成指针问题,其实是整数/浮点参与了条件判断

这个错误不一定是空指针解引用。比如int x; if (x > 0) { ... },x未初始化就参与比较,Valgrind就会报出来。64位系统上size 8大概率是longsize_t或指针类型,但根源是“用了没赋值的变量”,不是内存损坏。

  • 结构体成员未显式初始化(尤其含uint64_tdouble等非零默认值的字段)
  • new T[]不会调用构造函数,T是POD类型时整个数组都是未初始化的
  • memset(&obj, 0, sizeof(obj))看似清零,但对含虚函数或非平凡构造的类是UB,Valgrind可能后续报一堆衍生错误

Linux下跑Valgrind卡死或报Can't attach to process?检查ptrace权限和seccomp

容器环境(Docker/K8s)或启用了seccomp策略的系统,会禁止ptrace系统调用,而Valgrind依赖它注入检测逻辑。此时不是你的代码有问题,而是运行环境拦住了工具本身。

意兔-AI漫画相机
意兔-AI漫画相机

照片变漫画手绘,做周边好物

下载

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

  • Docker启动时加--cap-add=SYS_PTRACE,否则valgrind ./a.out直接失败
  • 某些发行版(如Ubuntu 22.04+)默认开启kernel.yama.ptrace_scope=2,需临时改回0sudo sysctl -w kernel.yama.ptrace_scope=0
  • 如果程序自己调用了prctl(PR_SET_NO_NEW_PRIVS, 1)seccomp过滤了ptrace,Valgrind必然失败,只能换到开发机上跑

Valgrind的报错行号和上下文基本可靠,但它的“推测”能力有限——比如把两次不同malloc的块搞混,或在内联函数展开后丢失原始调用链。真遇到绕不开的误报,优先怀疑是否用了__builtin_assume__attribute__((may_alias))这类打破别名规则的写法。

热门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

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

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

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号