资讯专栏INFORMATION COLUMN

【C语言】玩转递归——学好递归,你需要掌握的知识!

Donne / 3751人阅读

摘要:所以,递归在编程中同样是很重要的一个知识点。举个例子用递归实现求第个斐波那契数。总结起来四个字大事化小继续举斐波那契数的例子三递归是怎样运行的我们通过一道题目来讲解。


前言

在一定的时间、空间限制下,人的体力有限,思维力也有限,递归思维对实践最有用的指导,就是把脑力集中于定义问题这个关键点上,不用去找解题的过程。定义(问题)即解决(问题),定义即解决! 让大问题变成规模更小的问题并立即获得解决,以此作为基础,让我们轻松解决函数本身定义的问题。所以,递归在编程中同样是很重要的一个知识点。


提示:以下是本篇文章正文内容

一、递归是什么?

先来看一下定义:

程序调用自身的编程技巧称为递归( recursion)。

简单来说,就是在一个函数里面调用函数自己本身。

举个例子:
用递归实现求第n个斐波那契数。

int Fib(int n){	if (n <= 2)		return 1;	else		return Fib(n - 1) + Fib(n - 2);}int main(){	//斐波那契数 1 1 2 3 5 8 13 21 34 51....,除前两位外,后一个数的值等于前两位相加	int n = 0;	printf("请输入你要查找的斐波那契数:");	scanf("%d", &n);	int ret=Fib(n);	printf("你好,你要需要的值是:%d/n", ret);	return 0;}

所以,我们可以看到,所谓递归,其实就是一个函数里面调用函数自己本身。具体怎样调用的,我们下面再讲。

二、 递归的两个必要条件

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

分析之后,我们可以得出两个点

1、结束条件
2、逼近条件

我们在使用递归的时候,需要满足这两个条件。

总结起来四个字——大事化小

继续举斐波那契数的例子:

三、递归是怎样运行的

我们通过一道题目来讲解。

题目: 递归实现n的k次方
内容: 编写一个函数实现n的k次方,使用递归实现。

【解决思路】
运用递归思路,我们只要找到递归结束条件和逼近条件。通过分析,我们可以画出下面这幅图。

【代码实现】

#include double power(int n,int k){	if (k< 0)	{		k = -k;		return 1 / (n*power(n, k - 1));	}	else if (k == 0)		return 1;	else if (k>0)	{		return n * power(n, k - 1);	}}int main(){	int n = 0;	int k = 0;	printf("请输入一个整数:");	scanf("%d", &n);	printf("请输入要求的次方数:");	scanf("%d", &k);	double ret=power(n,k);	printf("%1f/n", ret);	return 0;}

【画图详解递归思路】

通过图解,发现思路,我们** 存在限制条件k,当满足这个限制条件的时候,递归便不再继续。
每次递归调用之后越来越接近这个限制条件。**之后输出的时候就反过来回去。

这个就是递归的思路。

四、迭代与递归

不知大家有没有认真思考过上面的求斐波那契数的代码,它有什么问题?


如果我们这里求的是第50个斐波那契数呢?大家可以运行一下代码。可以发现,电脑运行了好久好久才算出结果,费时间。
如果求第10000个斐波那契数呢?程序就会崩溃。

为什么呢?
我们发现上面求斐波那契数的 Fib 函数 在调用的过程中很多计算其实在一直重复。
因为我们在调用这个函数的时候,除前两位外,后一个数的值等于前两位相加。这就导致了我们不断重复计算

如图:
我们可以看到,由于前两个数相加等于后一个数,前两个数相加等于后一个数,所以我们会不断产生重复的计算。就会造成计算量非常大,效率极低

那我们如何改进呢?
我们程序存东西的时候,存放在栈区。
如图:

在调试 例子中的Fib函数的时候,如果你的参数比较大,那就会报错: `stack overflow(栈溢出) 这样的信息。
系统分配给程序的栈空间是有限的,但是如果出现了死循环,或者(死递归),这样有可能导致一直开辟栈空间,最终产生栈空间耗尽的情况,这样的现象我们称为栈溢出。

那如何解决上述的问题:

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

这里我们介绍迭代

什么是迭代呢?
【概念】

迭代是重复反馈过程的活动,其目的通常是为了逼近所需目标或结果。每一次对过程的重复称为一次“迭代”,而每一次迭代得到的结果会作为下一次迭代的初始值。

借用网上的图片来说明(侵删)

目前对于c语言来说,迭代可以简单认为是循环结构。

那么我们如何用迭代的方式求斐波那契数呢?
【代码如下】

int Fib(int n){	int a = 1;	int b = 1;	int c = 1;	while (n>2)	{		c = a + b;//求出c的值		a = b;//a赋值给b,也就是a作为b的值		b = c;//b赋值给c,也就是b作为c的值		n--;	}	return c;}int main(){	// 1 1 2 3 5 8 13 21 34 55,除前两位外,后一个数的值等于前两位相加	int n = 0;	printf("请输入你要查找的斐波那契数:");	scanf("%d", &n);	int ret = Fib(n);	printf("你好,你要需要的值是:%d/n", ret);	return 0;}

这样,我们算很大的数都能一下子算出来了,虽然不能保证正确,因为栈溢出了,但是效率很快。

五、递归与迭代的比较

我们用一个表格来分析:
【注意】

  1. 许多问题是以递归的形式进行解释的,这只是因为它比非递归的形式更为清晰。
  2. 但是这些问题的迭代实现往往比递归实现效率更高,虽然代码的可读性稍微差些。
  3. 当一个问题相当复杂,难以用迭代实现时,此时递归实现的简洁性便可以补偿它所带来的运行时开销。

六、 什么时候用递归

什么时候用递归呢?

(1)当解决一个问题时,递归和非递归都可以使用,且没有明显问题,那就可以使用递归
(2)当解决一个问题时,递归写起来很简单,非递归比较复杂,且递归没有明显问题,那就用递归
(3)如果说,用递归解决问题,写起来简单,但是有明显问题,那就不能使用递归


最后


以上内容是通过本人学习的理解和网上资料的整理梳理出来的递归与迭代的一些内容,有错漏之处,还请各位多多包涵与指出,共同进步,共同成长!

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/119667.html

相关文章

  • 如何学好c语言

    摘要:第二条军规必须画图理解,内存布局语言是一门偏底层的语言,可以直接操作访问内存的所以我们应该清楚知道,写出的代码所对应的内存布局。如果想学好语言,三条军规势在必行最后,关于学好语言我想说的也就到这里了,感谢你的观看。 目录 一.讲这个主题的原因 二.关于选择问题 三.具体学习方法 一.为什么要...

    xingpingz 评论0 收藏0
  • 如何系统地自学 Python?

    摘要:这里推荐一本书源码剖析源码剖析豆瓣这本书把源码中最核心的部分,给出了详细的阐释,不过阅读此书需要对语言内存模型和指针有着很好的理解。   是否非常想学好 Python,一方面被琐事纠缠,一直没能动手,另一方面,担心学习成本太高,心里默默敲着退堂鼓?   幸运的是,Python 是一门初学者友好的编程语言,想要完全掌握它,你不必花上太多的时间和精力。   Python 的设计哲学之一就是...

    zgbgx 评论0 收藏0
  • 软件开发入门学习个人看法。

    摘要:兴趣最后该说说的就是兴趣问题如果你能对它真正感兴趣如果要从事软件开发又没兴趣的话赶紧先培养兴趣去对看技术资料就想别人看武侠小说看球赛一样的话再配合上面提到的几点踏实先专后广基础扎实相信在这一行多少是可以做点东西出来的   踏实   偶然在网上看到《由C#风潮想起的-给初学编程者的忠告》一文. 其中一个角度:避免浮躁,倡导踏实的学习方法,我是很认同的,但总觉该文作者标题-给初学编程者的忠...

    wh469012917 评论0 收藏0

发表评论

0条评论

Donne

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<