初阶C语言:函数递归

简介: C语言知识点函数递归的详细介绍,让你轻松上手函数递归,递归虽好,可不要一直用递归喔!
函数递归是一种解决问题的好办法,但是能想到函数递归的解决方法也并不是轻而易举,还是得多思考思考,搞清楚函数递归的基本原理以及技巧就可以很好的使用递归来解决问题

紫蓝色几何渐变科技互联网微信公众号封面 (1).gif

1.什么是递归

程序调用自身的编程技巧称为递归( recursion)。 递归做为一种算法在程序设计语言中广泛应用。 一个过程或函数在其定义或说明中有直接或间接 调用自身的 一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解, 递归策略 只需少量的程序就可描述出解题过程所需要的多次重复计算,大大地减少了程序的代码量。 递归的主要思考方式在于:把大事化小 简单的说递归就是在一个函数中再调用这个函数

2.递归的两个必要条件

1.存在限制条件,当满足这个限制条件的时候,递归便不再继续。 2. 每次递归调用之后越来越接近这个限制条。

递归最常见的的错误就是死递归,举一个很简单的例子:

#include <stdio.h>intmain()
{
printf("hehe\n");
main();   //会导致栈溢出return0;
}

在讲static关键字时提到过栈区,和静态区,今天再来看一下

2.1小练习1

接受一个整型值(无符号),按照顺序打印它的每一位。 例如: 输入:1234,输出 1 2 3 4

假设要实现上述题目的代码,按照现在所学的C语言知识看似只能用函数递归,那该怎么用呢?给大家用代码演示一下,看不懂也不要紧,我也会给大家剖析函数递归的代码

代码演示:

#include <stdio.h>voidprint(intnum)
{
if (num>9)
    {
print(num/10);
    }
printf("%d ", num%10);
}
intmain()
{
intnum=0;
scanf("%d", &num);
print(num);
return0;
}

这段代码现在看不懂也没有关系,我们来分析一下,以输入123为例

代码进入 print 函数之后,进行判断:这时的num为123,123>9所以进入的条件语句,进入条件语句之后,执行print(num/10),这时还来不及执行下面的 printf 函数,然后再次进入 print 函数 进入 print 函数之后,这时的num为12,12>9所以又会进入条件语句,然后再次调用 print 函数,这时同样也没有机会执行下面的打印函数,然后再次进入 print 函数 进入print函数之后,这时的num为1,1<9因此不会满足条件,不会进入条件语句 这时代码再往下走遇到了打印函数 printf ,这时就要打印了,打印1%10,也就是打印1,这时③这个 print 函数就执行完了,这时屏幕上也就打印了1,然后代码就往回返,到了②的 print 函数,因为②的 print 函数还没有执行完,这时就要执行②中的 print 函数 返回上来时就要紧接着执行②中没有执行的代码了,这时就会打印num%10,也就是12%10也就是在屏幕上打印了2,然后这时②的 print 函数就执行完了,代码再往回返 代码返回到①的 print 函数就要执行它还没有执行完的代码,这时屏幕上就打印123%10,也就是3,这时全部的代码就都执行完了,屏幕上也就打印出了1 2 3 要注意,在代码执行到print(num/10)时就会再次调用这个函数,也就是说会进入一个新的print函数,这就表明这个条件语句还没有执行完,并不会执行条件语句下面的语句

2.2小练习2

编写函数不允许创建临时变量,使用函数递归求字符串的长度

我们知道有一个函数strlen专门用来求字符串的长度,那我们也来实现一下,首先得知道字符串的长度是统计\0之前的字符个数,因此不能将\0包含进去,还要求使用递归,这时就可以将求出的作为函数的返回值返回给主函数

代码演示:

#include <stdio.h>intmy_strlen(constchar*str)
{
if (*str!='\0')   //不能统计\0    {
return1+my_strlen(str+1);
    }
return0;
}
intmain()
{
charstr[] ="abc";
intlen=my_strlen(str);
printf("%d", len);
return0;
}

接下来对这段代码来进行一个解读

练习完这两个代码,应该对函数递归有了初步的认识了,函数递归的特点就是大事化小,可以看到使用函数递归写出的代码简洁

3.递归与迭代

C语言递归和迭代(循环)的相同之处在于,它们都是用来解决 重复性问题 的编程技术。 不同之处 在于,递归是一种分而治之的思想,它将一个复杂的问题分解成若干个规模较小的相同问题,而迭代则是一种重复执行某一操作的方法,它不断地重复执行某一操作,直到满足某一条件为止。

3.1练习

求n的阶乘 分别使用递归和迭代 1!  1 2!  2 * 1 3!  3 * 2 * 1 4!  4 * 3 * 2 * 1 ......
//递归#include <stdio.h>intfactorial(intn)
{
if (n<=1)
return1;  //1的阶乘就是1所以返回1elsereturnn*factorial(n-1);
}
intmain()
{
intn=0;
scanf("%d", &n);
intret=factorial(n);
printf("%d的阶乘为: %d", n, ret);
return0;
}

求第n个斐波那契数 (不考虑溢出)

要求斐波那契数首先得知道斐波那契数列是什么

1 1 2 3 5 8 13  21 34 55....

像这样的数列被叫做斐波那契数列

关于这个数列的规律就是从第三项开始,每一项都等于前两项之和。即:F(n)=F(n-1)+F(n-2),其中n>=3,F(1)=1,F(2)=1。

代码演示:

#include <stdio.h>intfib(intn)
{
if (n<=2)
    {
return1;
    }
elsereturnfib(n-1) +fib(n-2);
}
intmain()
{
intn=0;
scanf("%d", &n);
intret=fib(n);
printf("第%d个斐波那契数列是:%d", n, ret);
return0;
}

但是上述代码都有问题

  • 在使用factorial来计算10000的阶乘时(不考虑正确与否),程序会崩溃
  • 在使用fib来计算第50个斐波那契数列时会很慢

这是为什么呢?

因为fib在调用时进行了大量的重复运算,所以会导致在计算50个斐波那契数列时会非常的耗时间

我们可以看一下fib函数在调用时会重复计算多少次第3个斐波那契数列

代码演示:

#include <stdio.h>intcount=0;
intfib(intn)
{
//统计第三个斐波那契数列的重复个数if (n==3)
count++;
if (n<=2)
return1;
elsereturnfib(n-1) +fib(n-2);
}
intmain()
{
intn=0;
scanf("%d", &n);
intret=fib(n);
printf("第%d个斐波那契数列是:%d\n", n, ret);
printf("count = %d\n", count);
return0;
}

在计算第35个斐波那契数列时,就单纯的计算第3个斐波那契数列都整整重复计算了3524578次

那么该怎么改进上述两个代码呢?

可以使用非递归 (迭代) 的方法来实现

//n的阶乘//迭代n的阶乘#include <stdio.h>intfactorial(intn)
{
inti=0;
intnum=1;
for (i=1; i<=n; i++)
    {
num*=i;
    }
returnnum;
}
intmain()
{
intn=0;
scanf("%d", &n);
intret=factorial(n);
printf("%d的阶乘为: %d", n, ret);
return0;
}
//第n个斐波那契数列//迭代#include <stdio.h>intfib(intn)
{
inta=1;
intb=1;
intc=1;
while (n>=3)
    {
c=a+b;
a=b;
b=c;
n--;
    }
returnc;
}
intmain()
{
intn=0;
scanf("%d", &n);
intret=fib(n);
printf("第%d个斐波那契数列是:%d\n", n, ret);
return0;
}

Tip:

1. 许多问题是以递归的形式进行解释的,这只是因为它比非递归的形式更为清晰。 2. 但是这些问题的迭代实现往往比递归实现效率更高,虽然代码的可读性稍微差些。 3. 当一个问题相当复杂,难以用迭代实现时,此时递归实现的简洁性便可以补偿它所带来的运行时开销 函数递归经典例题(自主研究) 青蛙跳台阶 汉诺塔


关于函数递归的相关知识就分享到这里,感谢大家支持!

目录
相关文章
|
1月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
37 3
|
1月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
20 2
|
1月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
66 16
|
1月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
52 1
|
1月前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
61 24
|
1月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
63 23
|
1月前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
68 15
|
4月前
|
C语言
C语言函数返回值详解
本文详细解析了C语言中函数返回值的概念与应用。从函数的基本定义入手,深入探讨了不同类型返回值的作用及意义,并提供了实用的编程示例,帮助读者更好地理解和使用函数返回值。通过本文,你将掌握如何有效利用返回值优化代码结构与功能实现。
|
8月前
|
存储 C语言
C语言的函数返回值和指针
C|函数返回值(区分各类值)和指针(区分各类存储空间)的细节
|
9月前
|
存储 C语言
C语言中向函数传递值和从函数返回值的技术解析
C语言中向函数传递值和从函数返回值的技术解析
96 0

热门文章

最新文章