【C语言之函数栈帧】(动态图—巨细)一文带你了解局部变量随机值及栈区上的函数调用

news2025/1/10 10:27:48

🚩纸上得来终觉浅, 绝知此事要躬行。
🌟主页:June-Frost
🚀专栏:C语言

局部变量为什么是随机值?函数是如何调用的?

✉️ 该篇将使用该编译器,通过介绍栈帧的创建和销毁来深入了解局部变量和函数调用的一些细节。


目录:

    • 寄存器
    • 大致空间分布
    • 具体细节
    • 结语

寄存器

寄存器是CPU内部用来存放数据的一些小型存储区域,用来暂时存放参与运算的数据和运算结果。

先了解一下这几个寄存器:

eax:

  • 存储函数调用时返回值的地址
  • 存储线程级别的全局变量、堆栈指针等数据
  • 作为命令和参数传递时的参数值

ebx:

  • 存储基址或偏移量,用于访问数组或结构体中的元素
  • 存储线程级别的全局变量、堆栈指针等数据

ecx:

  • 存储计算结果或临时变量的地址
  • 存储线程级别的全局变量、堆栈指针等数据

edx:

  • 存储动态链接库或共享库的入口地址
  • 存储线程级别的全局变量、堆栈指针等数据

主角:

ebp:

  • 存储函数调用时的临时变量的地址
  • 存储函数调用时返回值的地址(在函数调用前由操作系统维护)
  • 存储栈帧中的基址

esp:

  • 存储当前栈帧的顶部指针,即下一个将要被压入栈中的内存地址
  • 存储函数调用时的临时变量的地址(在函数调用前由编译器维护)
  • 存储调用堆栈的回溯信息

ebp和esp 这两个寄存器中存放的是地址,用来维护函数栈帧。
每一次函数的调用,都会在栈区创建一块空间。

正在调用哪个函数,这两个寄存器就会去维护哪块空间。


大致空间分布

对于函数栈帧,通过一个例子来具体讲解:
代码为:

#include<stdio.h>
int Add(int x, int y)
{
	int z = 0;
	z = x + y;
	return z;
}
//空行
int main()
{
	int a = 10;
	int b = 20;
	int c = 0;
	c = Add(a, b);
	printf("%d\n", c);y
	return 0;
}

在这个代码中,都知道是main函数在调用Add函数
但是main函数谁在调用?
通过调用堆栈来观察:

📘这说明 是mainCRTSTartup() 调用了_tmainCRTSTartup() , _tmainCRTSTartup() 调用了 main(…),main(…)调用了Add(int x, int y)。

那么栈区的大致分布就是:


具体细节

我们将上述代码转到反汇编:

先对这部分代码进行详解:
在将要使用这部分main函数的时候,说明_tmainCRTSTartup() 的栈帧就已经创建完成了。


第一步:

push指令就是将指定的值压入函数的栈中。此处要将ebp的值压栈。
push       ebp

在执行第一步之前是这样的:

执行后为:

可以见到,栈区从高到低使用,esp的值减少了,意味着它确实指向了新压入的空间。

观察内存

说明确实压入的是ebp的值。


第二步:

mov    ebp,esp
将esp的值赋给ebp


第三步:

sub    esp,0E4h
sub指令是减法操作,这里相当于esp-0E4h,0E4h是八进制形式,也就是十进制228,所以为esp-228.

在这步,就成功创造了main函数的栈帧。


第四,五,六步:


这几步都是压栈。

监视:

内存:


第七,八,九,十步:

将这几步显示符号名:

lea指令用于将一个值从内存中复制到寄存器中,它的全称是"Load Effective Address",意思是"加载有效地址"。

这里相当于给edi加载了一个值,那个值其实就是没压栈前的esp的值0x008FFAC0 。

mov      ecx,39h
mov      eax,0CCCCCCCCh
这里都是赋值,将39h赋值给ecx,0CCCCCCCCh赋值给eax。

rep stos      dword ptr es:[edi]

word是2个字节,dword就是double word,是4个字节,这个操作就是将edi之后(包括自己)的39个dword赋为eax的值。
这里最终的效果其实就是将main函数的栈帧空间里的内存全部初始化为cc。


之后三步:

mov指令,将0Ah(10)赋值给[ebp-8]的那块空间,将14h(20)赋值给[ebp-14h]的空间,将0赋值给[ebp-20h]的空间。

初始化后才有确定值,说明如果没有初始化,那么对应的内存空间就是cc cc cc cc, 如果直接使用这种空间,那么对应的值就是随机值。


之后四步

这步是将[ebp-14h]空间的值赋给eax,再将eax压栈,将[ebp-8]空间的值赋给ecx,再将ecx压栈。

这里也反映了一个情况:函数的参数式从右向左传参的。


下一步:

当执行call指令的时候,就是调用函数,会出现两个效果,执行F11,跳转至该界面:

而且在内存上,会储存下一条指令的地址:

效果:


对上述界面再F11,就会进入Add函数。

Add函数内前面的这些汇编,和main有异曲同工之妙。都是在创建栈帧。
所以我将直接用动图直接展示这些步骤:



接下来,就需要找到[ebp - 8]的位置,并赋值0。


然后,找到[ebp + 8] 和 [ebp + 0Ch], 0Ch即12 。 先将[ebp + 8]对应的值赋给eax,再将[ebp + 0Ch]的值加在eax上,这样就完成了两数的相加,之后再将eax的值赋给[ebp - 8]对应的空间。



这里将 [ebp - 8]的值给了eax 。(变量z在出了函数体后,变量就被释放了,为了保存下这个值,编译器将该值放入寄存器中,就可以保证这个值的存在)。


pop用于将栈顶元素弹出到寄存器中 。栈顶原本就是dei的值,弹出后放入edi,这时栈顶就是esi的值,弹出后放入esi, 对于ebx同理。

接下来,将ebp的值给esp。



这两步骤很重要:
这时候栈顶ebp(main)【 这个数据其实就是之前main函数的ebp】

将栈顶数据弹出后再放入ebp其实就找回了这个位置。


ret指令用于将函数的返回地址压入栈中,以便在下一次调用时使用。简单的讲,该处就是会把call指令的下一条指令的地址弹出,并跳转到那里。

这时的栈帧空间就是这样的:


随着Add函数的调用完成,形参就可以释放了。

这里将esp+8后,栈帧空间内就不再管理那两个形参。

然后将eax的值赋值给[ebp - 20h] 。相当于c接收了返回值。

到此为止,其实一个函数栈帧的创建与销毁就完成了。

结语

文章到这里就结束了,本小白才疏学浅,如果文章存在问题,还请大佬们多多指出。💖💖💖

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/635218.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

计算机网络开荒3-传输层

文章目录 一、传输层概述1.1 网络层 vs 传输层 二、多路复用 多路分用三、UDP3.1 RDT3.1.1 Rdt3.1.1.1 Rdt1.03.1.1.2 Rdt2.03.1.1.3 Rdt2.13.1.1.4 Rdt2.23.11.5 Rdt 3.0 四、滑动窗口协议4.1 流水线机制4.1.2 滑动窗口协议GBNSR 五、TCP5.1 可靠数据传输5.1.1 RTT和超时 5.2 …

Vue中如何进行图表绘制

Vue中如何进行图表绘制 数据可视化是Web应用中非常重要的一部分&#xff0c;其中图表绘制是其中的重要环节。Vue作为一款流行的前端框架&#xff0c;提供了很多优秀的图表库&#xff0c;以满足不同业务场景下的需求。本文将介绍如何在Vue中进行图表绘制&#xff0c;包括使用Vu…

MM32F3273G8P火龙果开发板MindSDK开发教程4 - 滴嗒定时器Systick的配置

MM32F3273G8P火龙果开发板MindSDK开发教程4 - 滴嗒定时器Systick的配置 1、Systick寄存器 Systick是ARM内核的一个外设&#xff0c;所以在不同芯片的代码上移植比较方便&#xff0c;他总共有4个寄存器&#xff0c; 从Systick定义中可以看到&#xff1a; typedef struct {__I…

一文看懂Java中的锁

阅读本文你可以获得 Synchronized、ReentrantLock、ReentrantReadWriteLock、StampedLock、Condition、Semaphore、CountDownLatch、CyclicBarrier、JMM、Volatile、Happens-Before。 全文共16000字左右&#xff08;包含示例代码&#xff09;、欢迎收藏、在看、转发分批食用 一…

基于粒子群优化算法的配电网光伏储能双层优化配置模型[IEEE33节点](选址定容)(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

【MySQL 数据库】11、学习 MySQL 中的【锁】

目录 一、锁的概述与分类二、全局锁&#xff08;全库数据备份&#xff09;三、表级锁(1) 表锁(2) 元数据锁&#xff08;Meta Data Lock&#xff09;(3) 意向锁 四、行级锁(1) 行锁(2) 间隙锁&临键锁 一、锁的概述与分类 锁是计算机协调多个进程或线程并发访问某一资源的机…

Whistle(基于 Node 实现的跨平台抓包调试工具)的使用

Whistle(基于 Node 实现的跨平台抓包调试工具)的使用 基于Node实现的跨平台抓包调试工具 可以劫持网络请求&#xff0c;并进行请求和响应的修改&#xff0c;来提高我们的开发调试效率 1.一键安装(装包/证书) npm i -g whistle && w2 start --init 证书的问题 安装…

[论文阅读] (31)李沐老师视频学习——4.研究的艺术·理由、论据和担保

《娜璋带你读论文》系列主要是督促自己阅读优秀论文及听取学术讲座&#xff0c;并分享给大家&#xff0c;希望您喜欢。由于作者的英文水平和学术能力不高&#xff0c;需要不断提升&#xff0c;所以还请大家批评指正&#xff0c;非常欢迎大家给我留言评论&#xff0c;学术路上期…

5.2 清洗数据

5.2 清洗数据 5.2.1 检测与处理重复值1、记录重复 drop_duplicates()2、特征重复 equals() 5.2.2 检测与处理缺失值 isnull()、notnull()1、 删除法 dropna()2、替换法 fillna()3、 插值法 5.2.3 检测与处理异常值1、3σ原则2、箱线图 5.2.4 任务实现&#xff08;wei&#xff0…

学习HCIP的day.12

目录 MPLS&#xff1a;多协议标签交换 一、协议的解释和意义 二、工作过程 1、控制层面&#xff1a; 2、数据层面&#xff1a; 三、标签号 四、MPLS的次末跳 五、MPLS的配置 六、使用mpls解决BGP的路由黑洞 七、MPLS VPN 八、配置&#xff1a; 1、ISP部分 MPLS&am…

python+pyqt制作的可最小化到托盘的桌面图形应用代码实例

本篇文章主要讲解使用python、pyqt制作的可以最小化到托盘的桌面图形应用实例。 日期:2023年6月11日 作者:任聪聪 效果演示 说明:实现桌面应用显示窗口,关闭窗口缩小到托盘,点击托盘显示窗口,邮件图标显示退出按钮,点击退出按钮即可关闭应用。 动态演示: 实际情况:…

postgresql 服务的启动操作,不再是DBA的专属

专栏内容&#xff1a;postgresql个人主页&#xff1a;我的主页座右铭&#xff1a;天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物&#xff0e; 目录 前言 服务架构概述 服务启动流程 前提 流程 集群创建 集群介绍 数据库服务配置 数据…

202317读书笔记|《心寂犹似远山火:斋藤茂吉短歌300》——茫茫心海里,孤帆与谁同

202317读书笔记&#xff5c;《心寂犹似远山火&#xff1a;斋藤茂吉短歌300》——茫茫心海里&#xff0c;孤帆与谁同 很高兴周五这一天&#xff0c;之前很粉俳句的时候订阅的书都在今天都上架了&#xff0c;可以一饱眼福了。短歌是日本和歌一种诗体&#xff0c;是由三十一音节组…

YOLOv5/v7 添加注意力机制,30多种模块分析③,GCN模块,DAN模块

目录 一、注意力机制介绍1、什么是注意力机制&#xff1f;2、注意力机制的分类3、注意力机制的核心 二、GCN 模块1、GCN 模块的原理2、实验结果3、应用示例 三、DAN模块1、DAN模块的原理2、实验结果3、应用示例 大家好&#xff0c;我是哪吒。 &#x1f3c6;本文收录于&#xf…

扫雷——C语言实现

扫雷 文章目录 扫雷实现代码什么是扫雷基本功能实现显示选择菜单定义几个二维数组&#xff1f;确定数组大小初始化数组布置地雷打印展示数组排查地雷记录指定区域周围地雷的个数判断排雷成功排查地雷实现代码 基本功能的实现代码和效果展示 拓展功能简化游戏界面改变字体颜色实…

[创业之路-73] :如何判断一个公司或团队是熵减:凝聚力强、上下一心,还是,熵增:一盘散沙、乌合之众?

目录 前言&#xff1a; 一盘散沙、乌合之众&#xff1a; 凝聚力强、上下一心&#xff1a; 一、股权结构与利益分配 一盘散沙、乌合之众 凝聚力强、上下一心 二、组织架构与岗位职责 一盘散沙、乌合之众 凝聚力强、上下一心 三、战略目标 一盘散沙、乌合之众 凝聚力…

碳排放预测模型 | Python实现基于MLP多层感知机的碳排放预测模型(预测未来发展趋势)

文章目录 效果一览文章概述研究内容环境准备源码设计学习总结参考资料效果一览 ![1](https://img-blog.csdnimg.cn/34c113bde2 文章概述 碳排放预测模型 | Python实现基于MLP多层感知机的碳排放预测模型(预测未来发展趋势) 研究内容 这是数据集的链接:https://github.com/…

【伏羲八卦图】(PythonMatlab实现)

目录 1 与达尔文对话 2 与老子对话 2.1 Python实现 2.2 Matlab实现 1 与达尔文对话 140年前&#xff0c;1858年7月1日&#xff0c;达尔文在英伦岛发表了自己有关自然选择的杰出论文。他提出&#xff0c;生物的发展规律是物竞天择。经过物竞&#xff0c;自然界选择并存留最具…

【CAD】【动态块】CAD设置动态块

文章目录 1 CAD的动态块及应用2 块的相关概念2.1 块的相关命令2.1.1 创建块BLOCK2.1.2 插入块INSERT2.1.3 编辑块BEDIT2.1.4 重命名块RENAME 2.2 CAD组&#xff08;group&#xff09;和块&#xff08;block&#xff09;的区别2.3 “块”&#xff08;block&#xff09;和“写块”…

CMake学习(6): 打印日志信息及宏定义

1. message 打印日志 介绍CMake中的调试打印命令Message, 可以为用户显示一条消息&#xff0c;并在终端上打印显示。假如&#xff0c;我们通过File命令对文件进行了搜索&#xff0c;但是不能确定搜索到的文件就是我们需要的文件。此时&#xff0c;可以利用message将搜索到的变…