热门标签 | 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语言底层逻辑,是非常重要的哈!!!

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

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

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

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

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

下期见哈!!!



推荐阅读
  • 本文介绍如何利用栈数据结构在C++中判断字符串中的括号是否匹配。通过顺序栈和链栈两种方式实现,并详细解释了算法的核心思想和具体实现步骤。 ... [详细]
  • 本文详细介绍了8051系列微控制器的中断系统,特别是C51编译器中interrupt和using关键字的作用及其使用方法。通过深入分析这两个关键字的功能,帮助开发者更好地理解和优化中断程序的设计。 ... [详细]
  • 解析SQL查询结果的排序问题及其解决方案
    本文探讨了为什么某些SQL查询返回的数据集未能按预期顺序排列,并提供了详细的解决方案,帮助开发者理解并解决这一常见问题。 ... [详细]
  • C语言基础入门:7个经典小程序助你快速掌握编程技巧
    本文精选了7个经典的C语言小程序,旨在帮助初学者快速掌握编程基础。通过这些程序的实践,你将更深入地理解C语言的核心概念和语法结构。 ... [详细]
  • 本文探讨了如何使用自增和自减运算符遍历二维数组中的元素。通过实例详细解释了指针与二维数组结合使用的正确方法,并解答了常见的错误用法。 ... [详细]
  • 本文详细介绍如何在Linux系统中配置SSH密钥对,以实现从一台主机到另一台主机的无密码登录。内容涵盖密钥对生成、公钥分发及权限设置等关键步骤。 ... [详细]
  • 树链问题的优化解法:深度优先搜索与质因数分解
    本文介绍了一种通过深度优先搜索(DFS)和质因数分解来解决最长树链问题的方法。我们通过枚举树链上的最大公约数(GCD),将所有节点按其质因子分类,并计算每个类别的最长链,最终求得全局最长链。 ... [详细]
  • 深入理解Lucene搜索机制
    本文旨在帮助读者全面掌握Lucene搜索的编写步骤、核心API及其应用。通过详细解析Lucene的基本查询和查询解析器的使用方法,结合架构图和代码示例,带领读者深入了解Lucene搜索的工作流程。 ... [详细]
  • 异常要理解Java异常处理是如何工作的,需要掌握一下三种异常类型:检查性异常:最具代表性的检查性异常是用户错误或问题引起的异常ÿ ... [详细]
  • Hadoop发行版本选择指南:技术解析与应用实践
    本文详细介绍了Hadoop的不同发行版本及其特点,帮助读者根据实际需求选择最合适的Hadoop版本。内容涵盖Apache Hadoop、Cloudera CDH等主流版本的特性及应用场景。 ... [详细]
  • 采用IKE方式建立IPsec安全隧道
    一、【组网和实验环境】按如上的接口ip先作配置,再作ipsec的相关配置,配置文本见文章最后本文实验采用的交换机是H3C模拟器,下载地址如 ... [详细]
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • 主板IO用W83627THG,用VC如何取得CPU温度,系统温度,CPU风扇转速,VBat的电压. ... [详细]
  • 本文详细介绍了Grand Central Dispatch (GCD) 的核心概念和使用方法,探讨了任务队列、同步与异步执行以及常见的死锁问题。通过具体示例和代码片段,帮助开发者更好地理解和应用GCD进行多线程开发。 ... [详细]
  • 本文介绍如何从字符串中移除大写、小写、特殊、数字和非数字字符,并提供了多种编程语言的实现示例。 ... [详细]
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社区 版权所有