c语言基础知识帮助理解(函数递归详解)

这篇具有很好参考价值的文章主要介绍了c语言基础知识帮助理解(函数递归详解)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

"从前有座山,山里有座庙,庙里有个老和尚和一个小和尚。有一天老和尚对小和尚说:“从前有座山.山里有座庙,庙里有个老和尚和一个小和尚,有一天老和尚对小和尚说:“从前有座山.山里有座庙,庙里有个老和尚和一个小和尚......" (虽能体现递归特点,但又不是递归)


目录

1.什么是递归

2.递归的两个必要条件

 3.事例和讲解

 4.递归原理讲解

 5.递归弊端

5.1利用计算斐波那契数列来引入

5.2如何改进 


1.什么是递归

当一个函数在其定义中调用自身的过程称为递归。递归是一种强大的编程技巧,可以解决许多问题,特别是那些可以被分解为相同问题的子问题的情况

递归的主要思考方式在于:把大事化小

在C语言中,函数递归的基本原理是将一个大问题分解为一个或多个更小的问题,然后通过调用自身来解决这些更小的问题,直到达到基本情况,即不再需要递归调用的情况 


2.递归的两个必要条件

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

试想如若没有限制条件,那便无限循环,真就成了上面的“从前有座山 山里有座庙” 的故事了。


 3.事例和讲解

用递归来实现求n的阶乘(factorial) :

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
int factorial(int n)
{
	if (n > 1)//递归终止的条件
	{
		return n * factorial(n - 1);
	}
	else
	{
		return 1;
	}
}

int main()//递归实现求n的阶乘
{
	int n = 0;
	scanf("%d", &n);
	int n_factorial = factorial(n);
	printf("%d", n_factorial);
	return 0;
}

c语言基础知识帮助理解(函数递归详解),算法,c语言,数据结构,学习

下面是一个示例的递归树,用于说明计算4的阶乘的过程:

4! = 4 * 3!

|

3! = 3 * 2!

|

2! = 2 * 1!

|

1! = 1

c语言基础知识帮助理解(函数递归详解),算法,c语言,数据结构,学习

下面讲解一下思路:

  • 最基本便是:n!=n*(n-1)!——让我们有了返回n * factorial(n - 1)的想法
  • 当n=1时,便是终止条件——返回1 ,使用factorial(n - 1)也是为了更加靠近n=1的终止条件

 4.递归原理讲解

当一个函数被调用时,会在内存中分配一个称为函数栈帧(Function Stack Frame)的数据结构。函数栈帧用于存储函数的局部变量、函数的参数、函数的返回地址等信息。

下面是一个示意图,展示了函数栈帧的结构:

c语言基础知识帮助理解(函数递归详解),算法,c语言,数据结构,学习

  •  Return Address:保存着函数调用结束后需要返回的地址,即函数调用的下一条指令的地址。当函数执行完毕后,CPU会根据该地址跳转到正确的位置继续执行
  • Previous Stack Frame:指向调用函数的函数栈帧的地址,用于在函数返回时恢复调用函数的上下文
  • Local Variables:存储函数的局部变量。每个函数栈帧都有自己的一块内存空间,用于存储局部变量的值
  • Parameters:存储函数的参数。参数在函数调用时被传递给函数,并存储在函数栈帧中

在递归函数中,每次递归调用都会生成一个新的函数栈帧,这些函数栈帧会按照一定的顺序依次排列在内存中:先调用的函数先进入栈中,后销毁

 c语言基础知识帮助理解(函数递归详解),算法,c语言,数据结构,学习


 5.递归弊端

5.1利用计算斐波那契数列来引入

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

int main()
{
	int n = 3;
	printf("第三个斐波那契数%d\n", Fib(n));

	int x = 50;
	printf("第五十个斐波那契数%d", Fib(x));
	return 0;
}

c语言基础知识帮助理解(函数递归详解),算法,c语言,数据结构,学习

 我们会发现,第三个斐波纳契数立马就出来了,而第五十个迟迟不出来。

这就暴露了问题:

  • 这个函数的时候如果我们要计算第 50 个斐波那契数字的时候特别耗费时间
  • 函数求 10000 的阶乘(不考虑结果的正确性),程序会崩溃(我已经为各位试过了,不要再试了),也就是栈溢出了

这也说明:有时递归的效率不是很高 ,而且不得不说有时递归代码的可读性是不及循环的

5.2如何改进 

 那我们如何改进呢?

  • 将递归改写成非递归。
  • 使用static对象替代 nonstatic 局部对象。在递归函数设计中,可以使用 static 对象替代nonstatic 局部对象(即栈对象),这不仅可以减少每次递归调用和返回时产生和释放 nonstatic 对象的开销,而且 static 对象还可以保存递归调用的中间状态,并且可为各个调用层所访问

下面便用循环来改写求斐波那契数列:

int Fib(int n)
{
	int n1 = 1;
	int n2 = 1;
	int result = 0;
	while (n > 2)
	{n--;
		result = n1 + n2;
		n1 = n2;
		n2 = result;
		
	}
	return result;
}

int main()
{
	int n = 3;
	printf("第三个斐波那契数%d\n", Fib(n));

	int x = 30;
	printf("第三十个斐波那契数%d", Fib(x));
	return 0;
}

c语言基础知识帮助理解(函数递归详解),算法,c语言,数据结构,学习

所以我们也不要一味地去使用递归在合适的代码里使用合适的方法才能让我们的编程水平更加出色。 


总之,递归是一项强大的编程技术,但在使用时需要注意栈溢出问题。通过合理的算法设计和对函数栈帧的了解,我们可以更好地应对递归问题,使代码更加健壮和可靠

这次的分享先到这里的,感谢大家的支持,下一次会总结数组相关的知识!!! 文章来源地址https://www.toymoban.com/news/detail-625816.html

到了这里,关于c语言基础知识帮助理解(函数递归详解)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C语言基础知识:函数中的参数与返回值

    目录 1.形式参数和实际参数 1.1形式参数 1.2实际参数 2.变量作为函数参数 3.数组作为函数参数 3.1数组元素作为函数参数 3.2一维数组名作为函数参数 3.3数组指针,即数组元素的地址作为函数参数 4.函数返回值 形参出现在被调函数当中,在整个函数体内都可以使用。形参在定义

    2024年02月04日
    浏览(51)
  • 音视频基础知识——素材理解

    素材是媒体内容生产中一切生产资料的集合,包括不限于视频、音频、图片、字幕等形式。 素材通过统一的协议把原始的数据有序组织起来,便于编辑与管理。比如一般的素材是由一个物理文件及其各类属性构成,在对素材进行剪辑处理时,不用改变源文件中的任何内容,通

    2024年02月01日
    浏览(54)
  • 理解FPGA的基础知识——逻辑电路

    FPGA (Field Programmable Gate Aray,现场可编程门阵列)是一种可通过重新编程来实现用户所需逻辑电路的半导体器件。为了便于大家理解FPGA的设计和结构,我们先来简要介绍一些逻辑电路的基础知识。 1.逻辑代数     逻辑代数中的变量称为逻辑变量,用大写字母表示。逻辑变量的取

    2024年02月09日
    浏览(43)
  • 深入理解 HTTP Authorization 头:基础知识

    在当今的互联网世界中,安全性贯穿于 web 应用的每个方面,HTTP Authorization 头的使用在这个过程中扮演着不可或缺的角色。它是 HTTP 请求中的一个重要部分,用来在客户端和服务器之间安全地传输认证信息。用途广泛,无论是浏览器还是其他客户端应用,都依赖它来验证用户

    2024年04月15日
    浏览(42)
  • 深入理解机器学习——关联规则挖掘:基础知识

    分类目录:《深入理解机器学习》总目录 许多商业企业在日复一日的运营中积聚了大量的数据。例如,食品商店的收银台每天都收集大量的顾客购物数据。下图给出一个这种数据的例子,通常称作购物篮事务(Market Basket Transaction)。表中每一行对应一个事务,包含一个唯一

    2024年02月10日
    浏览(42)
  • 音视频基础知识-时间戳的理解

    问题背景: 凡是和流媒体和音视频打交道,时间戳基本是一个必须深刻理解的概念。你会在各种各样的传输协议和封装格式中看到这个东西,而且表现形式还不一样。其次这个概念会涉及到音视频播放的同步问题,也会影响音视频播放的控制问题。前者说的是音画同步,后者

    2023年04月08日
    浏览(43)
  • 深入理解 C++ 语法:从基础知识到高级应用

    让我们将以下代码分解以更好地理解它: 示例 示例解释 第 1 行: #include iostream 是一个头文件库,它让我们可以使用输入和输出对象,比如 cout (在第 5 行使用)。头文件为 C++ 程序添加功能。 第 2 行: using namespace std 表示我们可以使用标准库中的对象和变量名称。 如果你

    2024年03月23日
    浏览(46)
  • MOS管基础知识:轻松理解MOS管工作原理

    MOS管 是一种利用电场效应来控制其电流大小的半导体三端器件,很多特性和应用方向都与三极管类似。这种器件不仅体积小、质量轻、耗电省、寿命长、而且还具有输入阻抗高、噪声低、热稳定性好、抗辐射能力强等优点,应用广泛,特别是在大规模的集成电路中。 根据导

    2024年02月08日
    浏览(42)
  • 区块链基础知识——哈希函数

    哈希函数是一种数学函数,可根据输入的内容输出其对应的哈希值。散列函数的输入具有任意长度,但输出始终具有固定的长度。 散列算法的核心是一个数学函数,该散列函数构成散列算法的一部分,根据输入的数据块进行运算,得到的结果作为散列值。散列算法涉及多轮散

    2024年02月11日
    浏览(50)
  • C++ 从零基础到入门(3)—— 函数基础知识

    目录 一、函数简介 1、函数的作用和目的 2、函数的基本概念 二、函数定义与声明 1、函数定义的语法和结构 2、函数原型 三、参数传递 1、值传递 2、引用传递 3、指针传递 4、参数传递的选择 5.引用传递与指针传递的区别 四、返回值 1、返回类型 2、返回语句 在 C++ 中,函数

    2024年01月21日
    浏览(47)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包