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

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

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

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

103.png


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

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


0.png


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

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


1.png


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

形参的空间)。


2.png3.png4.png


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


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


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


5.开文题目的解答

5.png


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


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


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


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


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


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


下期见哈!!!

相关文章
|
8月前
|
存储 安全 C语言
深度剖析c语言程序 -- 函数栈帧的创建和销毁(纯肝货)-2
深度剖析c语言程序 -- 函数栈帧的创建和销毁(纯肝货)-2
|
8月前
|
存储 编译器 C语言
深度剖析c语言程序 -- 函数栈帧的创建和销毁(纯肝货)-1
深度剖析c语言程序 -- 函数栈帧的创建和销毁(纯肝货)-1
105 0
|
5月前
|
存储 C语言
【C语言】——函数栈帧的创建与销毁
【C语言】——函数栈帧的创建与销毁
|
8月前
|
存储 编译器 C语言
C语言:底层剖析——函数栈帧的创建和销毁
C语言:底层剖析——函数栈帧的创建和销毁
|
8月前
|
存储 编译器 程序员
C语言之反汇编查看函数栈帧的创建与销毁(二)
C语言之反汇编查看函数栈帧的创建与销毁(二)
|
10天前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
48 23
|
10天前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
40 15
|
10天前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
50 24
|
6天前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
46 16
|
6天前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
17 3

热门文章

最新文章