栈与递归,函数调用的背后发生了什么?

目录

一,书上总是把栈与递归放到一起,这是为什么呢?

二,调用函数的背后

三,例子说明


一,书上总是把栈与递归放到一起,这是为什么呢?

      我们开始执行程序之前,系统都会开辟一个函数调用栈,这个栈用来保存各个函数在调用过程当中所必须保存的一些信息,比如参数,局部变量,返回地址。当然自己调用自己,也离不开函数调用栈。

二,调用函数的背后

      当在一个函数运行期间,调用另一个函数的时候,在被调用函数之前,系统需要先完成三件事:

①将所有的实参返回地址等信息传递给被调用函数保存。

②被调函数的局部变量是封闭存储区。

将控制转移到被调函数的入口。

      被调函数返回给调用函数之前,系统也会完成三件事:

保存被调函数的计算结果。

释放被调函数的数据区域。

依照被调函数保存的返回值地址,将控制权限转移给调用函数。

      这么来说,先调用的函数最后执行完毕,后调用的函数先执行完毕,符合栈先进后出,后进先出的特点。

三,例子说明

那在运行中是怎么回事,来看看这个例子。

用递归求n的阶乘:

#include<stdio.h>
int f(int n)
{
	if (1 == n)
		return 1;
	else
		return n * f(n - 1);
}

int main()
{
	int n;
	printf("输入n=");
	scanf("%d", &n);
	f(n);
	printf("%d的阶乘为%d", n, f(n));
	return 0;
}

大家都知道过程如下。那么在函数调用栈中,它的递归是怎样体现的?

函数调用栈:

      在递归中,每次函数调用时都会将当前函数的执行状态保存到一个新的函数调用栈帧中并将该栈帧压入函数调用栈。同时,调用栈还会记录下一条要执行的指令的位置作为返回地址,并将它压入栈顶。当递归函数完成其任务并准备返回时,最近的函数调用栈帧被弹出,并使用保存的返回地址来跳转回调用该函数的代码处,从上一次已完成的地方继续执行。这个过程持续进行直到所有递归调用均已完成,整个递归算法得以结束。

      在二中提到,当在一个函数运行期间,调用另一个函数的时候,将所有的实参,返回地址等信息传递给被调用函数保存。被调函数返回给调用函数之前系统会保存被调函数的计算结果,释放被调函数的数据区域。前者就为递,后者就为归。当f(1)被计算出来之后,就开始进行归的过程,f(1)=1,f(2)=2*1=2,f(3)=3*2=6,f(4)=4*6=24,f(5)=24*5=120。   

      在执行代码之前,系统会将一些其他必备的东西压入栈中,如图中最底层的......,但我们不关心这个。我们关注的是,main函数会把实参n=5压入栈中,这个n始终为5,这也就是为什么在其他函数里改变n,在main函数里的n值是不会变的。

      

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

脑子不好真君

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值