热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C语言从入门到入土(进阶篇)函数栈帧的创建和销毁讲解(不看必后悔系列)(超详细)

在文章开始之前,先给大家补充两点,第一个是文章里面好像有一个我说的8进制,其实是16进制哈,就那么一处地方,当然也可能改了咳咳。第二点就是里面16进制数后面有0开头h结尾,那是计算

在文章开始之前,先给大家补充两点,第一个是文章里面好像有一个我说的8进制,其实是16进制哈,就那么一处地方,当然也可能改了咳咳。第二点就是里面16进制数后面有0开头h结尾,那是计算机那么写的,我们不用管,还是按16进制看就行了。

接下来正文开始啦!


谁都不能阻挡你成为更优秀的人。


再再再啰嗦最后一句,因为非常详细,而且需要理解,所以文章看起来有点吃力也是正常的。

目录

函数栈帧的创建和销毁 

 1.基础知识​

2.main函数栈帧的创建

 3.函数的调用

 4.函数的返回

分界线:main函数栈帧的创建和Add函数栈帧的创建于销毁

5.开文题目的解答




函数栈帧的创建和销毁 

看完之后可以解决上面的疑问。

当然,我们今天讲的函数栈帧都是在栈上开辟的,只讨论局部变量。


 1.基础知识

 


2.main函数栈帧的创建

 

 

 

 

 

PS:栈区内存从高地址向低地址走,所以我们等下往上走。

 

 

这里我们就可以通过调试里面的内存去看esp的地址里面存放的内容就是ebp的地址 

 

 

PS:这里预开辟的空间是编译器决定的,我们自己也是不知道的。  

 

PS:当我们压栈的时候esp是会变的。还有上面的那三个值,后面会自己弹出去的。

 

 

 

PS:是从edi那个位置向下的那么多空间。是改成CCCCCCCC哈,上面讲错了。 

 

 

这上面的就是main函数栈帧的开辟。

接下来就是执行有效代码了。


 3.函数的调用

 

PS:这里我们就可以说到,大家之前遇到过打印随机值的情况吧,

也就是烫烫烫烫烫之类的,在我们没有给那一块空间赋值的时候,

那个地方(如此CCCCCCCC),打印出来就是那样。

这里ab离得有点远,有也有可能是挨着放的,取决于编译器。

后面c也和前面一样。

综上:局部变量a,b,c的创建就是先创建一个函数栈帧,然后在里面找到

一块空间,再把a,b,c放进去。

 

 

 

 

 上面两个动作就是传参,但是还没有讲完,所以现在感受不到。

 

 PS:上面都是按的F10哈,这里call我们要按F11,进入函数内部。

 

 

我们记住call指令的下一条指令是因为当我们执行完call指令之后会回来,

回来的时候我们就找到这一条指令然后往下执行。

PS:右边的esp也是要指向最上面的哈。

再按F11,这次就是真正的来到了Add的函数里面。

 

也就是为我们的Add函数准备栈帧。

下面是我们此时的main函数的栈帧哈:

 

 

 


PS:我们从上还可以找到,函数传参压栈是从右向左传的,
在这里也就是先压b,再压a。
这下面就是xyz的图:

下图就完全可以说明:形参是实参的临时拷贝:

我们开始是在main函数中有a和b,我们通过push a,b 压了上去,也就是x(a‘),y(b’),

这就是临时(栈区)拷贝(压栈)。所以改变形参不会影响实参。

我们函数传参的方式是函数还没有调用的时候,参数就已经过去了,就是压了两个参数,然后当

我们真正进入函数内部的时候我们就找回了之前压的两个参数。然后再计算,计算好的值再放进z里面。

以上就是函数的调用,然后下面开始讲函数的返回。


 4.函数的返回

 


PS:就相当于用一个全局的寄存器先把z保存起来,也就是说安全了(不会消失)。等回到
主函数的时候再把eax拿出来用就可以了。

 

 开始:

 

 PS:一开始栈顶上是main函数的ebp。

 

结束:

现在就又回到了main函数里面了。

我们上面pop完之后只是要我们的ebp和esp指向了main函数的栈顶和栈底(即main函数

的这一块空间又用我们的ebp和esp开始维护了)。

但是我们回到了main函数之后还是要从call的下一个地址开始执行

怎么回到那里呢(call的下一个地址):

我们的栈顶上除了有main函数的ebp之外,其实还有call指令的下一条指令

其实ret指令就是返回时pop(弹出)call指令的下一条指令

所以说我们当时在栈顶存放call指令的下一条指令就是为了执行完

函数之后还能直接回到这里。

下面这就是call的下一条指令,回来之后就是消除了形参(释放了

形参的空间)。


分界线:main函数栈帧的创建和Add函数栈帧的创建于销毁

以上就是main函数栈帧的创建和Add函数栈帧的创建于销毁,

至于main函数栈帧就销毁和Add的差不多,就不过多赘述了。


5.开文题目的解答

1.首先为函数分配好栈帧空间,然后初始化一部分空间后,然后给

局部变量在栈帧里面分配一点空间,这就是局部变量的创建。

2.因为不初始化的时候里面的随机值是我们放进去的。(CCCCCCCC)

3.当我们要调用函数的时候,其实在我们还没有调用函数的时候,

我们就已经开始push从右向左开始压栈,压进去了,当我们真正进入

函数内部的时候,通过指针的偏移量,找回了我们的形参,这就是函

数的形参以及他的使用。

4.形参确实是我们在压栈的时候开辟的空间,他和我们的实参只是值是

相同的,空间是独立的,所以形参是实参的一份临时拷贝,改变形参,

不会影响实参。

5.就上面演示的。

6.在调用函数之前我们就把call函数的下一条指令地址记住了(压进去了),

把ebp调用这个函数的上一个函数的栈帧edp存进去了,当我们函数

调用完要返回的时候弹出edp就能找到原始上一个函数的edp然后指针

往下走的时候就能找到esp的地址,回到我们的栈帧空间,然后我们记

住了call指令下一条指令的地址,当我们往回返的时候就可以跳转到call

指令的下一条指令地址,让我们函数调用可以返回。返回值是通过寄存

器的方式带回来的。

PS:可以理解为Add的形参创建是开辟在main函数的函数栈帧里面的,只是

说Add在使用的时候,通过地址的偏移找到了a,b的拷贝x,y。

终终终终终终终终于讲完了!!!!!!

作者也没有想到自己的第一篇《c语言进阶》篇居然是讲函数栈帧,但是对函数栈帧的理解是能更好地去理解c语言底层逻辑,是非常重要的哈!!!

今天的内容就到这里了哈!!!

要是认为作者有一点帮助你的话!

就来一个点赞加关注吧!!!当然订阅是更是求之不得!

最后的最后谢谢大家的观看!!!

你们的支持是作者写作的最大动力!!!

下期见哈!!!



推荐阅读
  • 如何高效学习鸿蒙操作系统:开发者指南
    本文探讨了开发者如何更有效地学习鸿蒙操作系统,提供了来自行业专家的建议,包括系统化学习方法、职业规划建议以及具体的开发技巧。 ... [详细]
  • Android 开发技巧:使用 AsyncTask 实现后台任务与 UI 交互
    本文详细介绍了如何在 Android 应用中利用 AsyncTask 来执行后台任务,并及时将任务进展反馈给用户界面,提高用户体验。 ... [详细]
  • 本文旨在探讨Swift中的Closure与Objective-C中的Block之间的区别与联系,通过定义、使用方式以及外部变量捕获等方面的比较,帮助开发者更好地理解这两种机制的特点及应用场景。 ... [详细]
  • 本文详细介绍了如何在PyQt5中创建简易对话框,包括对话框的基本结构、布局管理以及源代码实现。通过实例代码,展示了如何设置窗口部件、布局方式及对话框的基本操作。 ... [详细]
  • 题目概述:Sereja 拥有一个由 n 个整数组成的数组 a1, a2, ..., an。他计划执行 m 项操作,这些操作包括更新数组中的特定元素、增加数组中所有元素的值,以及查询数组中的特定元素。 ... [详细]
  • C/C++中一级指针的内存模型与示例代码
    本文通过一个具体的C/C++代码示例,详细解析了一级指针在内存中的布局和工作原理。包括了对不同类型的指针变量如何在内存中分配空间的讨论。 ... [详细]
  • 本文探讨了Linux环境下线程私有数据(Thread-Specific Data, TSD)的概念及其重要性,介绍了如何通过TSD技术避免多线程间全局变量冲突的问题,并提供了具体的实现方法和示例代码。 ... [详细]
  • C/C++ 应用程序的安装与卸载解决方案
    本文介绍了如何使用Inno Setup来创建C/C++应用程序的安装程序,包括自动检测并安装所需的运行库,确保应用能够顺利安装和卸载。 ... [详细]
  • 本报告记录了嵌入式软件设计课程中的第二次实验,主要探讨了使用KEIL V5开发环境和ST固件库进行GPIO控制及按键响应编程的方法。通过实际操作,加深了对嵌入式系统硬件接口编程的理解。 ... [详细]
  • 最近遇到了一个关于单链表的编程问题,这是来自福富公司的笔试题目。以往我通常使用C语言来解决这类问题,但这次决定尝试用Java来实现。该题目要求实现一个单链表,并完成特定的方法。 ... [详细]
  • 本文探讨了互联网服务提供商(ISP)如何可能篡改或插入用户请求的数据流,并提供了有效的技术手段来防止此类劫持行为,确保网络环境的安全与纯净。 ... [详细]
  • 探讨了一个包含纯虚函数的C++代码片段,分析了其中的语法错误及逻辑问题,并提出了修正方案。 ... [详细]
  • hlg_oj_1116_选美大赛这题是最长子序列,然后再求出路径就可以了。开始写的比较乱,用数组什么的,后来用了指针就好办了。现在把代码贴 ... [详细]
  • 在尝试加载支持推送通知的iOS应用程序的Ad Hoc构建时,遇到了‘no valid aps-environment entitlement found for application’的错误提示。本文将探讨此错误的原因及多种可能的解决方案。 ... [详细]
  • 本问题涉及在给定的无向图中寻找一个至少包含三个节点的环,该环上的节点不重复,并且环上所有边的长度之和最小。目标是找到并输出这个最小环的具体方案。 ... [详细]
author-avatar
rseu_813
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有