大家经常说起函数的调用,但是很多人并不清楚函数调用过程的具体实现是怎样的,今天在这里我将通过一个例子浅谈自己对函数调用的理解。
函数调用过程要为函数开辟栈空间,用于本次函数的调用中临时变量的保存、现场保护。这块栈空间我们称之为函数栈帧。
栈帧的维护需要我们了解 ebp 和 esp 两个寄存器,在函数调用的过程中这两个寄存器存放了维护这个栈的栈底和栈顶指针。ebp 存放了指向函数栈帧栈底的地址;esp 存放了指向函数栈帧栈顶的地址。
源代码如下:
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int Add(int x,int y)
{
int sum = 0;
sum = x + y;
return sum;
}
int main()
{
int a = 10;
int b = 20;
int ret = Add(a, b);
system("pause");
return 0;
}
进入调试,转到反汇编语言如下:
![](https://i-blog.csdnimg.cn/blog_migrate/0347b9c40907584bb19cb0cf715440b6.png)
1. 从main函数开始,我们先来看 main 函数栈帧的创建。
![](https://i-blog.csdnimg.cn/blog_migrate/efc701a32587c30765eebdc142fae7a2.png)
重复拷贝值,初始化,如下:
![](https://i-blog.csdnimg.cn/blog_migrate/26907b6fe5154f902d07b1fe76c6da5c.png)
2. Add 函数的调用,参数传递过程如下。
进入 call 指令后,继续调试,按 F11,跳转如下:
再按 F11 就进入 Add 函数的执行代码处。
按 F11 后跳转到 add 指令处。
以上是关于函数调用及栈帧创建和销毁的代码文字分析,下画图解释一下函数的调用及栈帧(运行时堆栈)的创建,如下。
![](https://i-blog.csdnimg.cn/blog_migrate/e96b9dfada5f19d70b7c106cc53138d6.png)
对此有什么不正确或者需要改进的地方,还希望各位提出。