【C语言】函数递归

本文介绍了递归的概念、其在编程中的应用,重点讨论了递归的限制条件(如停止条件和效率问题),以及如何通过迭代和循环结构来避免栈溢出和提高效率。
摘要由CSDN通过智能技术生成

目录

一、递归的含义

二、递归的限制条件

1.有停止条件

2.能逐渐接近停止条件

三、递归的缺陷

1.栈溢出

2.效率低下

四、递归缺陷的解决

1.栈溢出示例解决

2.效率低下示例解决


一、递归的含义

        递归其实使一种解决问题的方法,在C语言中,递归就是函数自己调用自己,递就是递推,归就是回归

#include<stdio.h>

int main()
{
    printf("调用了一次main函数\n");
    main();//main函数调用了自身
    return 0;
}

        注意:上面是一个错误的递归示例,该例子的目的是让读者形象地了解递归


二、递归的限制条件

1.有停止条件

        递归存在一个停止条件,当满足这个停止条件时,递归便不再继续

2.能逐渐接近停止条件

        每次递归调用之后能逐渐接近停止条件,最终使递归停止

#include<stdio.h>

//计算函数的阶乘
int Fact(int x)
{
	if (x <= 1)//停止条件
	{
		return 1;
	}
	else
	{
		return x * Fact(x - 1);//x每次减1,x逐渐接近停止条件(x==0)
	}
}


int main()
{
	printf("%d", Fact(5));//输出:120
	return 0;
}

        下面通过画图深入了解递归的实现逻辑


三、递归的缺陷

1.栈溢出

        在C语言中,每一次函数调用,都需要在为函数在栈区内申请一块内存空间,用来保存函数调用期间的各种局部变量,这块内存空间就叫做函数栈帧

        函数递归调用时,若函数不返回,则函数对应的栈帧空间就一直占用

        若函数递归层数太深,则会浪费过多栈帧空间,引起栈溢出(stack overflow)

#include<stdio.h>

//计算函数的阶乘
int Fact(int x)
{
	if (0 == x)//停止条件
	{
		return 1;
	}
	else
	{
		return x * Fact(x - 1);//x每次减1,x逐渐接近停止条件(x==0)
	}
}


int main()
{
	printf("%d", Fact(99999));//递归层数太多,栈溢出
	return 0;
}

        在vs中按F5调试可以观察到此现象

2.效率低下

        一方面,在调用递归函数时,会频繁经历进栈和出栈两个过程,这会导致额外的时间开销

        另一方面,当递归表达式为多项式时,可能会将浅层递归大量重复计算,使时间开销大大增加

#include<stdio.h>


//计算第x位斐波拉契数列的值
int Fib(int x)
{
	if (x <= 2)
	{
		return 1;
	}
	else
	{
		return Fib(x - 1) + Fib(x - 2);
	}
}


int main()
{
	int n = 0;
	scanf("%d", &n);
	printf("%d", Fib(n));
	return 0;
}

        上面例子中,当n为45时,运行程序会出现明显的延迟才得出结果,当n大于50时,程序在短时间内已无法算出结果

        下面我们来探究原因

#include<stdio.h>

int count = 0;

//计算第x位斐波拉契数列的值
int Fib(int x)
{
	if (3 == x)//统计第三位斐波拉契数列被计算的次数
	{
		count++;
	}

	if (x <= 2)
	{
		return 1;
	}
	else
	{
		return Fib(x - 1) + Fib(x - 2);
	}
}


int main()
{
	int n = 0;
	scanf("%d", &n);//输入:45
	printf("%d\n", Fib(n));//输出:1134903170
	printf("第三位被计算了%d次", count);//输出:第三位被计算了433494437次
	return 0;
}

        由上面代码可以看出,当n为45时,Fib(3)被计算了四亿多次,这是完全不必要的计算,增加了大量无用的时间消耗


四、递归缺陷的解决

        递归层数过多导致的问题,一般可以考虑用迭代思想来解决(也就是我们常用的循环结构)

1.栈溢出示例解决

#include<stdio.h>

int main()
{
	int n = 0;
	scanf("%d", &n);
	int fact = 1;
	//求n的阶乘
	for (int i = 1; i < n; i++)
	{
		fact *= i;
	}
	printf("%d", fact);
	return 0;
}

2.效率低下示例解决

#include<stdio.h>

int main()
{
	int a = 1;
	int b = 1;
	int ans = 1;
	int n = 0;
	scanf("%d", &n);
	//计算第n位斐波拉契数列
	while (n > 2)
	{
		ans = a + b;
		a = b;
		b = ans;
		n--;
	}
	printf("%d", ans);
	return 0;
}

最后,感谢大家观看!

  • 8
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值