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

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

递归介绍



递归:就是函数自己调用自己。 子问题须与原始问题为同样的事,或者更为简单;

递归通常可以简单的处理子问题,但是不一定是最好的。


20190816205520468.gif


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

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


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

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); 
  } 
}


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


20190816213919107.png


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


20190816214226543.png


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


递归求阶乘



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)=-------
  }
}


运行流程为这样:


20190816215612876.gif


递归求斐波那契



按照上述思想,我们假设求斐波那契设成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);
  }
}


其实它的调用流程为:


image.gif


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


递归解决汉诺塔



汉诺塔是经典递归问题:


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


20190816222147593.png


  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. 如果更多,那么将会爆炸式增长。


20190816222824744.gif


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

  • 当有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.


20190816233419823.png


此时剩下底下最大的,只能移动到B,move(A,B)


20190816233743451.png


那么你是否发现什么眉目了,只需原先的huannuo(n-1)相同操作从C—>B即可完成转移到B;那么我们的之前函数应该写成hannuo(n-1,A,C)但是又用到B,所以把B传进来hannuo(n-1,A,B,C)先表示为从n-1个从A(借助B执行若干操作)转到C


20190816234625872.png


  • 这一系列操作使得将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 回复bigsai赠送精美资料一份!

目录
相关文章
|
7月前
|
存储 监控 安全
企业上网监控系统中红黑树数据结构的 Python 算法实现与应用研究
企业上网监控系统需高效处理海量数据,传统数据结构存在性能瓶颈。红黑树通过自平衡机制,确保查找、插入、删除操作的时间复杂度稳定在 O(log n),适用于网络记录存储、设备信息维护及安全事件排序等场景。本文分析红黑树的理论基础、应用场景及 Python 实现,并探讨其在企业监控系统中的实践价值,提升系统性能与稳定性。
315 1
|
7月前
|
存储 监控 算法
基于跳表数据结构的企业局域网监控异常连接实时检测 C++ 算法研究
跳表(Skip List)是一种基于概率的数据结构,适用于企业局域网监控中海量连接记录的高效处理。其通过多层索引机制实现快速查找、插入和删除操作,时间复杂度为 $O(\log n)$,优于链表和平衡树。跳表在异常连接识别、黑名单管理和历史记录溯源等场景中表现出色,具备实现简单、支持范围查询等优势,是企业网络监控中动态数据管理的理想选择。
206 0
|
11月前
|
存储 算法 Java
算法系列之数据结构-二叉树
树是一种重要的非线性数据结构,广泛应用于各种算法和应用中。本文介绍了树的基本概念、常见类型(如二叉树、满二叉树、完全二叉树、平衡二叉树、B树等)及其在Java中的实现。通过递归方法实现了二叉树的前序、中序、后序和层次遍历,并展示了具体的代码示例和运行结果。掌握树结构有助于提高编程能力,优化算法设计。
363 10
 算法系列之数据结构-二叉树
|
11月前
|
算法 Java
算法系列之数据结构-Huffman树
Huffman树(哈夫曼树)又称最优二叉树,是一种带权路径长度最短的二叉树,常用于信息传输、数据压缩等方面。它的构造基于字符出现的频率,通过将频率较低的字符组合在一起,最终形成一棵树。在Huffman树中,每个叶节点代表一个字符,而每个字符的编码则是从根节点到叶节点的路径所对应的二进制序列。
355 3
 算法系列之数据结构-Huffman树
|
11月前
|
算法 Java
算法系列之数据结构-二叉搜索树
二叉查找树(Binary Search Tree,简称BST)是一种常用的数据结构,它能够高效地进行查找、插入和删除操作。二叉查找树的特点是,对于树中的每个节点,其左子树中的所有节点都小于该节点,而右子树中的所有节点都大于该节点。
482 22
|
存储 机器学习/深度学习 算法
C 408—《数据结构》算法题基础篇—链表(下)
408考研——《数据结构》算法题基础篇之链表(下)。
446 30
|
存储 算法 C语言
C 408—《数据结构》算法题基础篇—链表(上)
408考研——《数据结构》算法题基础篇之链表(上)。
622 25
|
存储 人工智能 算法
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
这篇文章详细介绍了Dijkstra和Floyd算法,这两种算法分别用于解决单源和多源最短路径问题,并且提供了Java语言的实现代码。
1047 3
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
|
机器学习/深度学习 存储 缓存
数据结构与算法学习十:排序算法介绍、时间频度、时间复杂度、常用时间复杂度介绍
文章主要介绍了排序算法的分类、时间复杂度的概念和计算方法,以及常见的时间复杂度级别,并简单提及了空间复杂度。
678 1
数据结构与算法学习十:排序算法介绍、时间频度、时间复杂度、常用时间复杂度介绍
|
机器学习/深度学习 存储 算法
【数据结构与算法基础】——算法复杂度
【数据结构与算法基础】——算法复杂度

热门文章

最新文章