一,递归是什么?
C语言的学习我们会遇到一个绕不开的话题,那就是函数递归。那么究竟什么是递归?
下面让我们从最简单的一串递归代码开始了解。
#include <stdio.h>
int main()
{
printf("hehe\n");
main();//main函数中⼜调⽤了main函数
return 0;
}
从上图我们可以看到main函数中又调用了main函数,其实这就是函数递归。‘
![](https://img-blog.csdnimg.cn/cd52026f6b634f61bbdb1b7de54460f1.png)
二, 递归的限制条件
递归在书写的时候,有2个必要条件:
• 递归存在限制条件,当满足这个限制条件的时候,递归便不再继续。
• 每次递归调⽤之后越来越接近这个限制条件。
在下⾯的例⼦中,我们逐步体会这2个限制条件。
三,递归举例
1.求n的阶乘
计算n的阶乘(不考虑溢出),n的阶乘就是1~n的数字累积相乘。
分析:
![](https://img-blog.csdnimg.cn/b949d2149e584059b15e226159ab30e0.png)
![](https://img-blog.csdnimg.cn/920dd442e5f7496299b5852d3cee49ef.png)
int Fact(int n)
{
if(n<=0)
return 1;
else
return n*Fact(n-1);
}
测试:
#include <stdio.h>
int Fact(int n)
{
if (n <= 0)
return 1;
else
return n * Fact(n - 1);
}
int main()
{
int n = 0;
scanf("%d", &n);
int ret = Fact(n);
printf("%d\n", ret);
return 0;
}
![](https://img-blog.csdnimg.cn/dd98c8f03d6d4355889b841a9d2aa66d.png)
![](https://img-blog.csdnimg.cn/3f2bfa7d0df14b00b9ba28304732be45.png)
2.顺序打印一个数的每一位
输入一个整数m,输出按照顺序打印这个数的每一位。
比如:
题目分析
1234%10就能得到4,然后1234/10
得到123,这就相当于去掉了4
然后继续对123%10,就得到了3,再除10去掉3,以此类推
不断的 %10 和 \10 操作,直到1234的每⼀位都得到;
但是这⾥有个问题就是得到的数字顺序是倒着的
Print(n)
如果n是1234,那表⽰为
Print(1234) //打印1234的每⼀位
其中1234中的4可以通过%10得到,那么
Print(1234)就可以拆分为两步:
1. Print(1234/10) //打印123的每⼀位
2. printf(1234%10) //打印4
完成上述2步,那就完成了1234每⼀位的打印
那么Print(123)⼜可以拆分为Print(123/10) + printf(123%10)
以此类推下去
就有:
Print(1234)
==>Print(123) + printf(4)
==>Print(12) + printf(3)
==>Print(1) + printf(2)
==>printf(1)
void Print(int n)
{
if(n>9)
{
Print(n/10);
}
printf("%d ", n%10);
}
int main()
{
int m = 0;
scanf("%d", &m);
Print(m);
return 0;
}
结果:
在这个解题的过程中,我们就是使⽤了⼤事化⼩的思路
把Print(1234) 打印1234每⼀位,拆解为⾸先Print(123)打印123的每⼀位,再打印得到的4
把Print(123) 打印123每⼀位,拆解为⾸先Print(12)打印12的每⼀位,再打印得到的3
直到Print打印的是⼀位数,直接打印就⾏。
图示推演:
四,迭代与递归
递归是⼀种很好的编程技巧,但是很多技巧⼀样,也是可能被误⽤的,就像举例1⼀样,看到推导的公式,很容易就被写成递归的形式:
int Fact(int n)
{
if(n<=0)
return 1;
else
return n*Fact(n-1);
}
Fact函数是可以产⽣正确的结果,但是在递归函数调⽤的过程中涉及⼀些运⾏时的开销。
在C语⾔中每⼀次函数调⽤,都要需要为本次函数调⽤在栈区申请⼀块内存空间来保存函数调⽤期间的各种局部变量的值,这块空间被称为运⾏时堆栈,或者函数栈帧。
函数不返回,函数对应的栈帧空间就⼀直占⽤,所以如果函数调用中存在递归调⽤的话,每⼀次递归函数调⽤都会开辟属于⾃⼰的栈帧空间,直到函数递归不再继续,开始回归,才逐层释放栈帧空间。
所以如果采⽤函数递归的⽅式完成代码,递归层次太深,就会浪费太多的栈帧空间,也可能引起栈溢出(stack overflow)的问题。 。
所以如果不想使⽤递归就得想其他的办法,通常就是迭代的⽅式(通常就是循环的⽅式)。
⽐如:计算n的阶乘,也是可以产⽣1~n的数字累计乘在⼀起的。
int Fact(int n)
{
int i = 0;
int ret = 1;
for(i=1; i<=n; i++)
{
ret *= i;
}
return ret;
}
上述代码就是使用迭代的方式编写的,不仅能完成任务,而且比递归效率高出不少。
事实上,我们看到的许多问题是以递归的形式进行解释的,这只是因为它⽐⾮递归的形式更加清晰,
但是这些问题的迭代实现往往⽐递归实现效率更⾼。
当⼀个问题⾮常复杂,难以使⽤迭代的⽅式实现时,此时递归实现的简洁性便可以补偿它所带来的运行时开销。
例如:求第n个斐波那契数
![](https://img-blog.csdnimg.cn/f988689c6c21497da576ca432da4344d.png)
int Fib(int n)
{
if(n<=2)
return 1;
else
return Fib(n-1)+Fib(n-2);
}
#include <stdio.h>
int main()
{
int n = 0;
scanf("%d", &n);
int ret = Fib(n);
printf("%d\n", ret);
return 0;
}
![](https://img-blog.csdnimg.cn/b2e5704d88ef47f285c250a21c3b3c93.png)
#include <stdio.h>
int count = 0;
int Fib(int n)
{
if (n == 3)
count++;//统计第3个斐波那契数被计算的次数 8 if(n<=2) 9 return 1;
if (n <= 2)
return 1;
else
return Fib(n - 1) + Fib(n - 2);
}
int main()
{
int n = 0;
scanf("%d", &n);
int ret = Fib(n);
printf("%d\n", ret);
printf("\ncount = %d\n", count);
return 0;
}
结果:
这⾥我们看到了,在计算第40个斐波那契数的时候,使⽤递归⽅式,第3个斐波那契数就被重复计算了39088169次,这些计算是⾮常冗余的。所以斐波那契数的计算,使⽤递归是非常不明智的,我们就得想迭代的⽅式解决。
我们知道斐波那契数的前2个数都1,然后前2个数相加就是第3个数,那么我们从前往后,从⼩到⼤计算就行了。
这样就有下⾯的代码:
int Fib(int n)
{
int a = 1;
int b = 1;
int c = 1;
while(n>2)
{
c = a+b;
a = b;
b = c;
n--;
}
return c;
}
通过这样迭代的方式去解决问题,效率就高出了不少。
总结:在思考问题时要选择最合适的方式,有时候一个问题递归能够一下子想出来代码怎么编写,但计算量过大时,递归就略显无力了,这时候就得试着使用迭代来解决问题。
故解决问题应选择最适合的方式。