当前位置: 首页 > news >正文

Valgrind Memcheck 全解析教程:6个程序说明基础内存错误

Valgrind 是一个强大的动态分析框架,其中的 memcheck 工具用于检测 C/C++ 程序中类型不定的内存错误,是基础级内存调试工具的重要选择。

本文将通过 6 段有意义的错误代码,全面讲解 memcheck 的检测原理和输出分析,进而帮助学习者托底基础。


一、memcheck 基本原理

Valgrind 使用一种动态代码插装技术(Dynamic Binary Instrumentation),保留所有内存分配/释放/操作记录,并对每一次访问进行约束性检查:

  • 访问是否在合法范围内
  • 访问的内容是否已被初始化
  • 是否对释放后的地址进行操作
  • 是否对非添配内存调用 free()

二、六种错误结构解析

在这里插入图片描述

1. 堆内存超级读取 + Use-After-Free

char *p = malloc(1);
*p = 'a';
char t = *(p + 1);     // 超级读取
free(p);
*p = 3;               // 释放后写入

Valgrind 输出

  • Invalid read of size 1 表示超级读
  • Invalid write of size 1 表示 Use-After-Free

分析malloc(1) 只有一个 byte,访问 p+1超范围。 free(p) 后再写 *p是释放后操作,是精准级检测值点。


2. 重复释放 (Double Free)

char *p = malloc(1);
*p = 'a';
free(p);
free(p);

Valgrind 输出

  • Invalid free() 指出上一次释放地址

分析: 重复释放将破坏 heap 内部结构,在 glibc 中可能导致 abort(),Valgrind 可精确抓出。


3. 非添配地址释放 (Invalid Free)

char p = 'a';
free(p);

Valgrind 输出

  • Invalid free() 指出试图释放的是 stack 地址

分析p 是一个普通变量,释放非堆内存是第一级的编程错误,Valgrind 可相当精准地检出。


4. 内存泄漏 (Memory Leak)

char *p = malloc(1);
*p = 'a';
// no free

Valgrind 输出

  • definitely lost: 1 bytes in 1 blocks

分析: 程序退出时 heap 中存在未释放内存块,Valgrind 会标记泄漏类型,并可通过 --leak-check=full 查看分配地点。


5. 未初始化内存读取 (Uninitialized Read)

char *p = malloc(1);
char t = *p;
printf("chat t = %c\n", t);
free(p);

Valgrind 输出

  • Conditional jump or move depends on uninitialised value(s)

分析: malloc 分配的内存是随机值,直接读取而未初始化,会导致打印出的内容非确定,Valgrind 较好地检测读操作是否在可信区域内。


6. 释放后读取 (Use-After-Free: Read)

char *p = malloc(1);
*p = 'a';
free(p);
char t = *p;     // 释放后读
printf("chat t = %c\n", t);

Valgrind 输出

  • Invalid read of size 1

分析: 释放后内存地址成为无效,再读取就是 Use-After-Free 错误,尽管访问成功,结果也是未知行为。


结论

valgrind --tool=memcheck 是分析程序内存问题的重要工具:

  • 它能检测 heap 区间的超级、未初始化、释放错误;
  • 较难检测 stack 超级或静态区超级;
  • 通过精确输出并配合 --track-origins=yes,可相当精精确确地保障基础级 C 编程的内存健康。
http://www.dtcms.com/a/288648.html

相关文章:

  • 判断一个数是否为质数方法
  • VSCode使用Jupyter完整指南配置机器学习环境
  • c#:TCP服务端管理类
  • 正点原子stm32F407学习笔记10——输入捕获实验
  • 2025 年科技革命时刻表:四大关键节点将如何重塑未来?
  • 内网后渗透攻击过程(实验环境)--3、横向攻击
  • SQL 调优第一步:EXPLAIN 关键字全解析
  • 【已解决】GitHub SSH 连接失败解决方案:Permission Denied (publickey) 错误修复指南
  • [Linux]进程 / PID
  • 30天打牢数模基础-决策树讲解
  • Linux入门篇学习——NFS 服务器的搭建和使用和开发板固件烧录
  • Spring Boot 第一天知识汇总
  • 【Java项目安全基石】登录认证实战:Session/Token/JWT用户校验机制深度解析
  • 相似度计算
  • 「Java案例」利用方法求反素数
  • Facebook 开源多季节性时间序列数据预测工具:Prophet 饱和预测 Saturating Forecasts
  • dynamic_cast的实现原理
  • Beamer-LaTeX学习(教程批注版)【6】
  • Elasticsearch 简化指南:GCP Google Compute Engine
  • GPT-4o mini TTS:领先的文本转语音技术
  • 随着GPT-5测试中泄露OpenAI 预计将很快发布 揭秘GPT-5冲击波:OpenAI如何颠覆AI战场,碾压谷歌和Claude?
  • prometheus 黑盒监控和docker检测
  • mysql第三次作业
  • 学习寄存器——GPIO(二)学习BSRR BRR ODR寄存器的原子性和在HAL库的应用
  • 【Go语言-Day 22】解耦与多态的基石:深入理解 Go 接口 (Interface) 的核心概念
  • 【详细笔记】两类曲线积分转换
  • 群组功能实现指南:从数据库设计到前后端交互,上班第二周
  • 【数据结构】揭秘二叉树与堆--用C语言实现堆
  • 人工智能之数学基础:随机实验、样本空间、随机事件
  • Docker Desktop 入门教程(Windows macOS)