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.pswp.cn/bicheng/89882.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/89882.shtml
英文地址,请注明出处:http://en.pswp.cn/bicheng/89882.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vue3 实现 Excel 文件导入导出功能

在Vue 3中实现Excel文件的导入和导出功能,你可以使用一些流行的JavaScript库,如SheetJS(也称为xlsx)来处理Excel文件。以下是实现这一功能的基本步骤:1. 安装SheetJS首先,你需要安装xlsx库。在你的Vue项目中…

CS231n-2017 Lecture2图像分类笔记

图像分类问题定义:在已有固定的分类标签集合的前提下,能够对输入的图像进行识别处理,从集合中找到该图像所对应的标签。对于计算机而言,图像并非直观的图像,而是一个的像素集合,对于每个像素,其…

Typecho博客Ajax评论功能实现全攻略

文章目录 Typecho实现Ajax评论功能的完整指南 引言 一、技术选型与准备工作 1.1 技术栈分析 1.2 环境准备 二、前端实现方案 2.1 基础HTML结构 2.2 JavaScript处理逻辑 三、后端处理实现 3.1 创建插件处理Ajax请求 3.2 错误处理增强 四、安全性考虑 4.1 CSRF防护 4.2 输入过滤 …

【计算机考研(408)- 数据结构】树与二叉树

树与二叉树 树的定义及相关概念 树是n(n≥0)个结点的有限集合,n 0时,称为空树,这是一种特殊情况。在任意一棵非空树中应满足: 1)有且仅有一个特定的称为根的结点。 2)当n > 1时…

MacOS:如何利用终端来操作用户

MacOS:如何利用终端来操作用户MacOS:如何利用终端来操作用户1. 创建用户并赋予管理员权限步骤:2. 取消用户的管理员权限解释:3. 查看组成员查看 admin 组成员:查看 users 组成员:4. 其他常见的用户管理命令…

基于SpringBoot+MyBatis+MySQL+VUE实现的医疗挂号管理系统(附源码+数据库+毕业论文+答辩PPT+项目部署视频教程+项目所需软件工具)

摘 要 在如今社会上,关于信息上面的处理,没有任何一个企业或者个人会忽视,如何让信息急速传递,并且归档储存查询,采用之前的纸张记录模式已经不符合当前使用要求了。所以,对医疗挂号信息管理的提升&#x…

学成在线项目

黑马程序员学成在线项目学习过程记录 解决跨域问题

Shell脚本-grep工具

一、前言在 Linux/Unix 系统中,grep 是一个非常强大且常用的文本搜索工具,它可以帮助我们快速从文件或标准输入中查找匹配特定模式的内容。无论是查看日志、调试脚本,还是进行自动化数据提取,grep 都扮演着至关重要的角色。本文将…

深入解析Ext2文件系统架构

要在硬盘上存储文件,必须先将硬盘格式化为特定类型的文件系统。文件系统的主要功能就是组织和管硬盘中的文件。在Linux系统中,最常见的文件系统是Ext2系列,其早期版本为Ext2,后续又发展出Ext3和Ext4。虽然Ext3和Ext4对Ext2进行了功…

商业秘密保护:从法律理论到企业实战

作者:邱戈龙、柯坚豪深圳商业秘密律师广东长昊律师事务所在商业竞争中,商业秘密就像企业的"隐形护城河"。从法律角度看,它的保护路径经历了三次重要升级:从最初的"合同约定"到后来的"财产保护"&…

AI产品经理面试宝典第36天:AI+旅游以及行业痛点相关面试题的指导

一、AI如何解决旅游行业核心痛点? 面试官提问: "请结合具体案例说明AI在旅游行业的应用价值,以及它如何解决传统旅游服务的痛点?" 你的回答: 以腾讯"一部手机游云南"为例,AI技术通过四大核心体系重构旅游体验: 数字身份体系:通过人脸识别与用户…

【conda】Linux系统中部署Conda环境

目录 一、安装 Miniconda 1.1 下载 Miniconda 安装脚本 1.2 运行安装脚本 1.3 初始化 Conda: 安装完成后,初始化 Conda 环境 1.4 验证安装 二、设置虚拟环境默认存放路径(可选) 三、conda创建虚拟环境 3.1 创建 Conda 环境…

Spring Boot 解决跨域问题

在 Spring Boot 中解决跨域问题(CORS)主要有三种常用方式,下面详细说明每种实现方法: 方案一:全局配置(推荐) 在配置类中实现 WebMvcConfigurer 接口,统一配置所有接口的跨域规则&am…

Softhub软件下载站实战开发(十九):软件信息展示

上一篇文章中我们上线了软件分离展示&#xff0c;本篇文章我们聚焦软件信息展示 软件列表信息展示 点击一级分类查询该分类下所以软件分类切换要有动画效果分页支持 核心实现 <transition-grouptag"div"class"software-grid"before-enter"before…

[HDLBits] Cs450/gshare

Branch direction predictor 分支方向预测器 A branch direction predictor generates taken/not-taken predictions of the direction of conditional branch instructions. It sits near the front of the processor pipeline, and is responsible for directing instructio…

[学习] 双边带调制 (DSB) 与单边带调制 (SSB) 深度对比

双边带调制 (DSB) 与单边带调制 (SSB) 深度对比 文章目录双边带调制 (DSB) 与单边带调制 (SSB) 深度对比**数学原理****调制表达式与频谱****时域特性****频域特性****Python 仿真代码****仿真结果分析****工程应用建议**数学原理 设基带信号为 m(t)m(t)m(t)&#xff08;带宽为…

Gitee 提交信息的规范

在使用 git push 命令将代码推送到 Gitee&#xff08;或任何 Git 平台&#xff09;时&#xff0c;引号中的信息通常指的是 提交信息&#xff08;Commit Message&#xff09;。提交信息是对本次代码修改的简要描述&#xff0c;规范的提交信息有助于团队协作和版本管理。 Gitee 提…

C 语言经典编程题实战:从基础算法到趣味问题全解析

在 C 语言学习过程中&#xff0c;通过实战编程题巩固知识点是非常有效的方式。本文整理了一系列经典 C 语言编程题&#xff0c;涵盖基础计算、逻辑判断、图形打印等多个维度&#xff0c;并附上完整代码与解析&#xff0c;适合初学者参考学习上机题1.计算n以内所有正奇数的和 ?…

Chapter 3 Design of Switching Power Regulators

Chapter 3 Design of Switching Power Regulators Power Management Techniques for Integrated Circuit Design by Ke-Horng Chen 这本书比较深, 简单介绍基本概念后, 就直接抛出大段公式和结论, 一章讲其他书几章内容, 适合有一定基础, 想进一步做电源系统的人查阅. 优点是不…

算法题(176):three states

审题&#xff1a; 本题需要我们找到最佳铺设道路&#xff0c;将三个国家联通起来&#xff0c;然后输出最佳铺设道路的铺设数量&#xff0c;若没有联通方法则输出-1 思路&#xff1a; 首先我们正面思考&#xff1a;只需从某个点出发然后搜索到三个国家即可&#xff0c;最后对比所…