C++遵循模块化的思想,将独立的功能会封装为函数实现,让代码逻辑更为清晰。那么了解函数调用时的堆栈变化就尤为重要。今天我们介绍函数堆栈调用等相关基础知识。
一、函数堆栈调用
系统会为每一个函数开辟一个栈帧,可以保存函数的变量等信息,是函数的活动空间,有时也称为栈区。
那么我们可以从汇编的角度看一下函数栈帧,去查看函数堆栈调用过程,这就涉及到基本的汇编指令,和寄存器的使用,我们列出基本的寄存器和汇编指令:
【1. 寄存器:】
- eax ebx ecx edx 用来存储数据
- ebp:栈底指针寄存器,指向栈帧底部
- esp:栈顶指针寄存器,指向栈帧顶部
【2. 基本指令:】
mov A,B //赋值指令,A=B,如果是地址表示A指向B执行的地址空间
push A //压栈,将A压入栈帧
call 下一行指令地址 //禁止相对位移调用指令,首先压入下一行指令地址,再跳转到调用函数内部
add A,B //累加操作,A+=B,地址就是下移(上低下高)
sub A,B //累减操作,A-=B,地址就是上移
lea A,B //移地址指令,和赋值指令类似,A地址指向B地址的位置
rep stos A //循环处理,将A指向的地址不断累加赋值,一般用于开辟的空间初始化。
ret //将栈顶指针指向的下一行指令地址出栈,表示调用函数结束,继续主函数的执行
ret x //比ret指令多一件事,将栈顶指针esp+x,表示下移清理形参空间
我们通过下面的函数来讲解整个函数函数堆栈调用过程:
#include<stdio.h>
int Sum(int left,int right)
{
int temp=0;
temp=left+right;
return tmp;
}
int main()
{
int a=10;
int b=20;
int rt=0;
rt=Sum(a,b);
printf("rt:%d\n",rt)