首页
学习
活动
专区
圈层
工具
发布
社区首页 >专栏 >函数的递归

函数的递归

作者头像
寻星探路
发布2025-12-17 19:48:37
发布2025-12-17 19:48:37
300
举报
文章被收录于专栏:CSDN博客CSDN博客

一、递归的含义

递归其实是⼀种解决问题的⽅法,在C语⾔中,递归就是函数⾃⼰调⽤⾃⼰。

我们可以用一个简单的C语言递归代码:

代码语言:javascript
复制
 #include <stdio.h>
 
 int main()
 {
     printf("hehe\n");
     main();//main函数中⼜调⽤了main函数 
     return 0;
 }

这就是一个简单的递归程序,程序在运行的过程中,经过main()会进入并开启下一轮的执行,不过这里只是为了演示递归,这个代码里面由于没有任何限制条件让它停下来,程序会陷入死循环,导致栈溢出。(储存空间有限,被占满后溢出)

1.递归的思想

把⼀个⼤型复杂问题层层转化为⼀个与原问题相似,但规模较⼩的⼦问题来求解;直到⼦问题不能再被拆分,递归就结束了。所以递归的思考⽅式就是把⼤事化⼩的过程。

递归中的递就是递推的意思,归就是回归的意思,接下来慢慢来体会。

2.递归的限制条件

递归在书写的时候,有2个必要条件:

(1)递归存在限制条件,当满⾜这个限制条件的时候,递归便不再继续。

(2)每次递归调⽤之后越来越接近这个限制条件。

二、递归的举例

1.求n的阶乘

(⼀个正整数的阶乘(factorial)是所有⼩于及等于该数的正整数的积,并且0的阶乘为1。 ⾃然数n的阶乘写作n!)

题⽬:计算n的阶乘(不考虑溢出),n的阶乘就是1~n的数字累积相乘。

1.1分析并实现代码

n的阶乘的公式: n ! = n∗(n−1)!

代码语言:javascript
复制
举例:
     5! = 5*4*3*2*1
     4! = 4*3*2*1 
所以:5! = 5*4!

从这个公式不难看出:如何把⼀个较⼤的问题,转换为⼀个与原问题相似,但规模较⼩的问题来求解的。

n的阶乘和n-1的阶乘是相似的问题,但是规模要少了n。有⼀种有特殊情况是:当n==0 的时候,n的阶乘是1,⽽其余n的阶乘都是可以通过上⾯的公式计算。

那我们就可以写出函数Fact求n的阶乘,假设Fact(n)就是求n的阶乘,那么Fact(n-1)就是求n-1的阶 乘,于是我们可以得到这样的代码:

代码语言:javascript
复制
 #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;
 }

运行发现,这个代码是符合我们的要求的(这里不考虑n太大的情况,n太大存在溢出)

1.2绘图分析

代码从左侧开始,沿着实线方向进行

2.顺序打印整数的每一位

输⼊⼀个整数m,按照顺序打印整数的每⼀位。

⽐如:

输⼊:1234 输出:1234

输⼊:520 输出:520

2.1分析和实现代码

如果n是⼀位数,n的每⼀位就是n⾃⼰

n是超过1位数的话,就得拆分每⼀位

(这里就不过多的解释了,不懂的可以看一下我之前的博客(三种循环),和里面讲到的逆序打印是类似的,只是多了一个递归)

代码语言:javascript
复制
 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⼀样,看到推导的公式,很容易就被写成递归的形式,但是在递归函数调用的过程中会涉及到一些运行时的开销。

在C语⾔中每⼀次函数调⽤,都需要为本次函数调⽤在内存的栈区,申请⼀块内存空间来保存函数调 ⽤期间的各种局部变量的值,这块空间被称为运⾏时堆栈,或者函数栈帧。

函数不返回,函数对应的栈帧空间就⼀直占⽤,所以如果函数调⽤中存在递归调⽤的话,每⼀次递归 函数调⽤都会开辟属于⾃⼰的栈帧空间,直到函数递归不再继续,开始回归,才逐层释放栈帧空间。 所以如果采⽤函数递归的⽅式完成代码,递归层次太深,就会浪费太多的栈帧空间,也可能引起栈溢 出(stackoverflow)的问题。

(函数栈帧后面也会有专门的博客讲到)

所以如果不想使⽤递归,就得想其他的办法,通常就是迭代的⽅式(通常就是循环的⽅式)。

⽐如:计算n的阶乘,也是可以产⽣1~n的数字累计乘在⼀起的。

代码语言:javascript
复制
 int Fact(int n)
 {
     int i = 0;
     int ret = 1;
     for(i = 1; i <= n; i++)
     {
         ret *= i;
     }
     return ret;
 }

上述代码是能够完成任务,并且效率是⽐递归的⽅式更好的。

事实上,我们看到的许多问题是以递归的形式进⾏解释的,这只是因为它⽐⾮递归的形式更加清晰, 但是这些问题的迭代实现往往⽐递归实现效率更⾼。

当⼀个问题⾮常复杂,难以使⽤迭代的⽅式实现时,此时递归实现的简洁性便可以补偿它所带来的运 ⾏时开销。

例:求第n个斐波那契数

(斐波那契数列:第一、二个数都是1,其余数字都等于其前两数之和)

看到这公式,很容易诱导我们将代码写成递归的形式,如下所示:

代码语言:javascript
复制
 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;
 }

当我们输入的结果较小时,可以得到我们想要的答案,但是,如果我们输入50,可以发现需要很久才能算出结果(甚至几分钟)

如上图所示,仅仅是计算50就要耗费大量的时间和精力,这就说明了递归写法真的非常低效!!!

我们不妨在上面的代码中增加一点功能,看一下有多少冗余的计算:

代码语言:javascript
复制
#include <stdio.h>
int count = 0;
 
int Fib(int n)
 {
     if(n == 3)
         count++;//统计第3个斐波那契数被计算的次数 
     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):

这⾥我们看到了,在计算第40个斐波那契数的时候,使⽤递归⽅式,第3个斐波那契数就被重复计算了 39088169次,这些计算是⾮常冗余的。所以斐波那契数的计算,使⽤递归是⾮常不明智的,我们就得 想迭代的⽅式解决。

代码语言:javascript
复制
 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;
 }

迭代的⽅式去实现这个代码,效率就要⾼出很多了。

(由于编译器有着数据上限,所以输出错误很正常,但是它的运行速度极快!)

 有时候,递归虽好,但是也会引⼊⼀些问题,所以我们⼀定不要迷恋递归,适可⽽⽌就好。

拓展学习:

(1)⻘蛙跳台阶问题

(2)汉诺塔问题

以上2个问题都可以使⽤递归很好的解决,有兴趣可以研究。

(当然,也可以看我后面的博客!!!)

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2025-12-09,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档