数据结构与算法—递归(阶乘、斐波那契、汉诺塔)

目录

递归介绍

递归:就是函数自己调用自己。 子问题须与原始问题为同样的事,或者更为简单;
递归通常可以简单的处理子问题,但是不一定是最好的。

对于递归要分清以下概念:

  • 自己调用自己
  • 递归通常不在意具体操作,只关心初始条件上下层的变化关系
  • 递归函数需要有临界停止点,即递归不能无限制的执行下去。通常这个点为必须经过的一个数。
  • 递归通常能被其他方案替代(栈、数组正向求)。

认识递归,递归函数通常简易但是对于初学者可能很难取理解它。拿一个递归函数来说。

static void digui()
{
	System.out.println("bigsai前");
	digui();
	System.out.println("bigsai后");
}

是一个递归吧?
不是正常递归没有结束条件,自己一致调用自己死循环
那正确的递归应该这样

static void digui(int time)
{
	if(time==0) {}//time==0不执行
	else {
		System.out.println("bigsai前time: "+time);
		digui(time-1);
		System.out.println("bigsai后time: "+time);
	}
}

对于这样一种递归,它的执行流程大致是这样的

所以,调用dugui(5)在控制台输出是这样的

那么,我想你对递归函数执行的流程应该有所了解了吧。

递归求阶乘

求 n!=n*(n-1)*-----*1=n!=n*(n-1)
所以阶乘的上下级的关系很容易找到。我们假设一个函数jiecheng(n)为求阶乘的函数。
这个阶乘,你可以这样命名:

int jiecheng(int n)
{
   int va=1;
   for(int i=n;i>0;i--)
   {
       va*=i;
   }
   return i;
}

但是你还可以简便这样:

static int jiecheng(int n)
{
	if(n==0)//0的阶乘为1
	{
		return 1;
	}
	else {
		return n*jiecheng(n-1);//return n*(n-1)*jiecheng(n-2)=-------
	}
}

运行流程为这样:

递归求斐波那契

按照上述思想,我们假设求斐波那契设成F(n);
首先,斐波那契的公式为:

  • F[n]=F[n-1]+F[n-2](n>=3,F[1]=1,F[2]=1)
  • 也就是除了n=1和2特殊以外,其他均是可以使用递推式。

那么递推实现的代码为:

static long F(int n)
{
	if(n==1||n==2) {return 1;}
	else {
		return F(n-1)+F(n-2);
	}
}

其实它的调用流程为:

当然,其效率虽然不高,可以打表优化,后面可能还会介绍矩阵快速幂优化!

递归解决汉诺塔

汉诺塔是经典递归问题:

相传在古印度圣庙中,有一种被称为汉诺塔(Hanoi)的游戏。该游戏是在一块铜板装置上,有三根杆(编号A、B、C),在A杆自下而上、由大到小按顺序放置64个金盘(如下图)。游戏的目标:把A杆上的金盘全部移到C杆上,并仍保持原有顺序叠好。操作规则:每次只能移动一个盘子,并且在移动过程中三根杆上都始终保持大盘在下,小盘在上,操作过程中盘子可以置于A、B、C任一杆上。

  1. 如果A只有一个(A->C)
  2. 如果A有两个(A->B),(A->C),(B->C)
  3. 如果A有三个(A->C),(A->B),(C->B),(A->C),(B->A),(B->C),(A->C).
  4. 如果更多,那么将会爆炸式增长。

可以发现每增加一步,其中的步骤会多很多。但是不妨这样想:

  • 当有1个要从A->C时,且已知移动方式。使用函数表示move(a->c)。同理其他move操作。
  • -------省略中间若干步骤不看,用递归思想看问题

分析:n个从a—>cn-1个a—>c有什么联系?(hannuo(n)—>hannuo(n-1)有啥关系)
假设有n个盘子

  • hannuo(n-1)之后n-1个盘子从A—>C.
  • 此时剩下底下最大的,只能移动到B,move(A,B)
  • 那么你是否发现什么眉目了,只需原先的huannuo(n-1)相同操作从C—>B即可完成转移到B;那么我们的之前函数应该写成hannuo(n-1,A,C)但是又用到B,所以把B传进来hannuo(n-1,A,B,C)先表示为从n-1个从A(借助B执行若干操作)转到C。
  • 这一系列操作使得将n个盘子从A—>B但是我们要的是A—>C才是需要的hannuo(n,A,B,C);那么我们只需要更改下hannuo(n-1,----)顺序就好啦!

经过上面分析,那么完整的操作为:

package 递归;
public class hannuota {
	static void move(char a,char b)
	{
		System.out.println("移动最上层的"+ a+ "到"+ b+ "\t");
	}
	static void hannuota(int n,char a,char b,char c)//主要分析每一大步对于下一步需要走的。
	{
		if(n==1) {move(a,c);}//从a移到c
		else
		{
			hannuota(n-1,a,c,b);//将n-1个从a借助c移到b
			move(a,c); //将第n(最后一个)从a移到c。
			hannuota(n-1,b,a,c);//再将n-1个从b借助a移到c
		}
	}
	public static void main(String[] args)
	{

		hannuota(5,‘a‘,‘b‘,‘c‘);
	}
}

总结

其实递归在某些场景的效率是很低下的。尤其是斐波那契.从图你就可以发现一个简单的操作有多次重复。因为它的递归调用俩个自己.那么它的递归的膨胀率是指数级别的,重复了大量相同计算。当然这种问题也有优化方案的:

  • 从前往后打表计算,采用类似动态规划的思想。从前往后考虑。比如斐波那契F[n]=F[n-1]+F[n-2];那么我用数组储存。从第三项开始F[3]=F[2]+F[1](均已知),再F[4]=F[3]+F[2]-----这样,时间复杂度是O(N),线性的。
  • 当然,对于阶乘那种递归虽然时间是没有减少,但是如果需要多次访问一个阶乘,那么可以采用同样思想(打表)解决问题。

最后,笔者能力有限,如果有描述不恰当还请指正,感谢前面动态图(未找到原作者)和汉诺塔动图开源作者isea533的开源作品。同时,如果有喜欢学习交流的欢迎关注笔者公众号:bigsai 回复数据结构赠送精美资料一份!

原文地址:https://www.cnblogs.com/bigsai/p/11371926.html

时间: 2024-10-26 02:57:50

数据结构与算法—递归(阶乘、斐波那契、汉诺塔)的相关文章

js递归阶乘斐波那契规律

例子 //阶乘 function getRes(n) { if(n == 1) {return 1;} return getRes(n-1) * n; } let a = getRes(6); console.log(a); //斐波那契 function feb(n) { if(n == 1 || n == 2) { return 1;} return feb(n-1) + feb(n-2); } let b = feb(3); console.log(b); 规律: 先找固定的 if(固定的

13、蛤蟆的数据结构笔记之十三栈的应用之栈与递归之斐波那契数列

13.蛤蟆的数据结构笔记之十三栈的应用之栈与递归之斐波那契数列 本篇名言:"人生不是一支短短的蜡烛,而是一支由我们暂时拿着的火炬,我们一定要把它燃得." 继续递归的斐波那契数列问题. 欢迎转载,转载请标明出处: 1.  斐波那契数列 斐波那契数列,又称黄金分割数列,指的是这样一个数列:0.1.1.2.3.5.8.13.21.--在数学上,斐波纳契数列以如下被以递归的方法定义:F(0)=0,F(1)=1,F(n)=F(n-1)+F(n-2)(n≥2,n∈N*)在现代物理.准晶体结构.化学

249 递归:概念,利用递归求1~n的阶乘,利用递归求斐波那契数列,利用递归遍历数据

6.1什么是递归 递归:如果一个函数在内部可以调用其本身,那么这个函数就是递归函数. 简单理解: 函数内部自己调用自己, 这个函数就是递归函数 注意:递归函数的作用和循环效果一样,由于递归很容易发生"栈溢出"错误(stack overflow),所以必须要加退出条件return. <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"&g

算法——动态规划篇——斐波那契数列

斐波那契数列,又称黄金分割数列,指的是这样一个数列:0.1.1.2.3.5.8.13.21.--在数学上,斐波纳契数列以如下被以递归的方法定义:F0=0,F1=1,Fn=F(n-1)+F(n-2)(n>=2,n∈N*)在现代物理.准晶体结构.化学等领域,斐波纳契数列都有直接的应用,为此,美国数学会从1960年代起出版了<斐波纳契数列>季刊,专门刊载这方面的研究成果. 以上内容来自百度百科.. 今天主要是想用动态规划的思想求解斐波那契数列,用来观察动态规划带来的优势,空间换时间,不重复求解

谨慎地使用递归之斐波那契递归实现的分析

[斐波那契函数的定义] 斐波那契数列,又称黄金分割数列,指的是这样一个数列:1.1.2.3.5.8.13.21.--在数学上,斐波纳契数列以如下被以递归的方法定义:F0=1,F1=1,Fn=F(n-1)+F(n-2)(n>=2,n∈N*). [用递归求解斐波那契函数的弊端] 斐波那契函数用递归实现如下面的代码: 计算斐波那契的自然递归程序效率时很低的,上面的程序虽然写法简单,看上去时递归的聪明的使用,但其实效率是及其低下的.特别是当n=40以后,效率衰减的特别明显.为了计算 fib( n ) ,

几年前做家教写的C教程(之三专讲了递归和斐波那契)

C语言学习宝典(3) 数组: 一维数组的定义: 类型说明符  数组名[常量表达式] 例如: int  a[10]; 说明:(1)数组名的命名规则和变量名相同,遵循标示符命名规则 (2)在定义数组时需要指定数组个数,即数组长度 (3)变量表达式中可以包括常量和符号常量,不能包含变量. 一维数组的应用:  数组名[下标] 一维数组的初始化:(1)在定义数组时对数组元素赋予初值 Int a[10]={0,1,2,3,4,5,6,7,8,9} (2)可以只给一部分元素赋值 Int a[10]={0,1,

【算法】还在用递归实现斐波那契数组,面试官一定会鄙视你到死

我记得在初学C语言的时候,大学老师经常会讲一些常见的数学问题及递归的使用,其中斐波那契数组的实现就是一定会被拿出来举例的.在后来工作中,面试做面试题的时候,也很大概率会出现编程实现斐波那契额数组算法.可以说,在我们编程道路上,编写程序实现斐波那契数组算法是每个程序员必定会做的一件事. 斐波那契数列指的是这样一个数列 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233,377,610,987,1597,2584,4181,6765,10946,17711,

Python递归及斐波那契数列

递归函数 在函数内部,可以调用其他函数.如果一个函数在内部调用自身本身,这个函数就是递归函数.举个例子,我们来计算阶乘 n! = 1 * 2 * 3 * ... * n,用函数 fact(n)表示,可以看出:fact(n) = n! = 1 * 2 * 3 * ... * (n-1) * n = (n-1)! * n = fact(n-1) * n所以,fact(n)可以表示为 n * fact(n-1),只有n=1时需要特殊处理.于是,fact(n)用递归的方式写出来就是: def fact(

算法笔记_001:斐波那契数的多种解法

本篇文章解决的问题来源于算法设计与分析课程的课堂作业,主要是运用多种方法来计算斐波那契数.具体问题及解法如下: 一.问题1: 问题描述:利用迭代算法寻找不超过编程环境能够支持的最大整数的斐波那契数是第几个斐波那契数.(Java: 231-1 for int, 263-1 for long) 解决方案:针对问题1,此处要使用迭代法来解决,具体实现代码如下: //用迭代法寻找编程环境支持的最大整数(int型)的斐波那契数是第几个斐波那契数 public static int max_int_iter