递归是什么
递归其实是⼀种解决问题的⽅法,在C语⾔中,递归就是函数⾃⼰调⽤⾃⼰。写⼀个史上最简单的C语⾔递归代码:
#include <stdio.h>
int main() {
printf("hehe\n");
main();
return 0;
}
上述就是⼀个简单的递归程序,只不过上⾯的递归只是为了演⽰递归的基本形式,不是为了解决问 题,代码最终也会陷⼊死递归,导致栈溢出(Stack overflow)。
递归的思想:
把⼀个⼤型复杂问题层层转化为⼀个与原问题相似,但规模较⼩的⼦问题来求解;直到⼦问题不能再被拆分,递归就结束了。所以递归的思考⽅式就是把⼤事化⼩的过程。递归中的递就是递推的意思,归就是回归的意思。
递归的限制条件
递归在书写时,有两个必要条件
递归存在限制条件,当满足这个限制条件的时候,递归不在继续。
每次递归调用之后越来越接近这个限制条件。
递归举例
求n的阶乘
int Fact(int n) {
if (n == 0) {
return 1;
}
else {
return n * Fact(n - 1);
}
}
int main() {
int sun = 0;
int a = 5;
sun = Fact(a);
printf("%d",sun);
}
这里不考虑n太大,太大会出现溢出。
图画演示

顺序打印一个整数的每一位
输入一个整数m,按照顺序打印整数的每一位
比如:输入1234 输出 1 2 3 4
自我分析一下,如何得到这个数的每一位,如果n是一位数,n的每一位就是自己,如果n超过一位,就要拆分每一位。1234%10就能得到4,然后1234/10得到123,就去掉了4,然后继续对123%10,得到了3,再除10去掉了3,以此类推就会得到每一位数字
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;
}
画图演示

递归与迭代
递归是⼀种很好的编程技巧,但是和很多技巧⼀样,也是可能被误⽤的,就像举例1⼀样,看到推导的 公式,很容易就被写成递归的形式:

int Fact(int n)
{
if(n==0)
return 1;
else
return n*Fact(n-1);
}
Fact函数是可以产⽣正确的结果,但是在递归函数调⽤的过程中涉及⼀些运⾏时的开销。在C语⾔中每⼀次函数调⽤,都需要为本次函数调⽤在内存的栈区,申请⼀块内存空间来保存函数调 ⽤期间的各种局部变量的值,这块空间被称为运⾏时堆栈,或者函数栈帧。函数不返回,函数对应的栈帧空间就⼀直占⽤,所以如果函数调⽤中存在递归调⽤的话,每⼀次递归 函数调⽤都会开辟属于⾃⼰的栈帧空间,直到函数递归不再继续,开始回归,才逐层释放栈帧空间。 所以如果采⽤函数递归的⽅式完成代码,递归层次太深,就会浪费太多的栈帧空间,也可能引起栈溢出(stackoverflow)的问题。
所以如果不想使⽤递归,就得想其他的办法,通常就是迭代的⽅式(通常就是循环的⽅式)。⽐如:计算n的阶乘,也是可以产⽣1~n的数字累计乘在⼀起的。
int Fact(int n) {
int i = 0;
int ret = 1;
for (i = 1;i <= n;i++) {
ret *= i;
}
return ret;
}
int main() {
int m = 0;
int n = 0;
scanf("%d", &m);
n = Fact(m);
printf("%d", n);
}
上述代码是能够完成任务,并且效率是⽐递归的⽅式更好的。事实上,我们看到的许多问题是以递归的形式进⾏解释的,这只是因为它⽐⾮递归的形式更加清晰, 但是这些问题的迭代实现往往⽐递归实现效率更⾼。当⼀个问题⾮常复杂,难以使⽤迭代的⽅式实现时,此时递归实现的简洁性便可以补偿它所带来的运⾏时开销。
求第n个斐波那契数
斐波那契数列(Fibonacci sequence),又称黄金分割数列,因数学家莱昂纳多·斐波那契(Leonardo Fibonacci)以兔子繁殖为例子而引入,故又称“兔子数列”。这个数列从第3项开始,每一项都等于前两项之和。
用递归方式实现
int FibDG(int n) {
if (n <= 2) {
return 1;
}
else {
return Fib(n - 1) + Fib(n - 2);
}
}
用迭代方式实现
int FibDD(int n) {
int a = 1;
int b = 1;
int c = 1;
if (n > 2) {
c = a + b;
a = b;
b = c;
n--;
}
return c;
}
迭代的⽅式去实现这个代码,效率就要⾼出很多了。有时候,递归虽好,但是也会引⼊⼀些问题,所以我们⼀定不要迷恋递归,适可⽽⽌就好。

3万+

被折叠的 条评论
为什么被折叠?



