递归汉诺塔《Blind-Stab》

2023-10-18 18:20
文章标签 递归 汉诺塔 blind stab

本文主要是介绍递归汉诺塔《Blind-Stab》,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

函数递归
从代码上来看 就是函数调用函数自己
递归有什么好处呢?可以用最简便的方式求解问题
递归有什么坏处呢?过于深的递归会导致栈溢出异常
1)写递归时,一定要先清楚合适开始归(递归到那一层就应该停止)
2)递归主要是将大的问题逐步化解为小的问题进而解决

直接上代码吧!

比较简单经典的递归解法问题比如说递归求和?

//递归方法求前100项的和
public class Demo {public static void main(String[] args) {System.out.println(sum(100));}public static int sum(int n) {if(n == 1) {                       //递归的判断条件return 1;}return n+sum(n-1);                 //递归}
}

 再比如说递归求斐波那契第n项

//递归求斐波那契前n项和
public class Demo {public static void main(String[] args) {System.out.println(fibo(2));}public static int fibo(int n) {if(n == 1 ||n == 0) {    //归的条件return 1;}return fibo(n-1)+fibo(n-2); //递}
}

再比如说汉诺塔问题

//汉诺塔移动盘子问题
public class HanNo {public static void main(String[] args){Scanner scanner = new Scanner(System.in);System.out.println("请输入盘子个数");int level = scanner.nextInt();            //输入盘子个数move("A","B","C",level);          //传入参数,(从哪里移动,中介底座,移到哪里,移几个)}public static void move(String from, String mid, String to,int level){if(level == 1){                    //只有一个盘子,那就直接打印移动啊System.out.println(from+"->"+to);}else{move(from,to,mid,level-1);      //将level - 1个盘子从A底座移动到B底座,空出最底下那个盘子System.out.println(from+"->"+to);//打印直接最后一个盘子可以直接移动move(mid,from,to,level-1);//陷入循环,移动level - 1个盘子,回到了开始的柑橘}}
}

关于汉诺塔这一题,我觉得很有必要细讲。这可能是递归的真谛,或者说最好的应用。

比如说输入3,也就是3个盘子,那么程序运行结果如上,也就是移法如上。这里from是从哪里移动,mid是需要借助的盘子,to是目的盘。关键是move函数,这里应该用点逆向思维和整体思维去理解递归。下面我们来分析一下这个盘子怎么去移动。

分析:

现在有三个底座A,B,C

我们解释一下构造的递归函数里的参数     move(args1,args2,args3,args4)我们要知道前三个参数代表的都是底座第四个代表的是要操作盘子的个数。

第一个参数args1,代表的是当前操作要移动盘子对应的底座。也就是从哪个底座移,也就是从args1处移。

第二个参数args2,代表的是需要借助的位置,其实就是中介盘。为了参数个数需要传进来的。其实没什么l用。

第三个参数args3,代表的是当前移动盘子的目的地。也就是移到哪个底座去,也就是移动到arg4底座去。

第四个参数args4,代表的是你想移动的盘子的个数

1,首先要将所有得盘子从A移动到C,那必然要先将第n个盘子取出放到C。

2,想要将第n个盘子从A到C那必须先将前面n-1个盘子先放到B,这样才可以移动C,所以再此之前我们得先写上

move(from,to,mid,level-1);

3,这个时候第n个盘子终于可以被移动了,所有我们来了一个

System.out.println(from+"->"+to);

4,好,刚才移动的第n个盘子已经到家了,那就不想她了,不管她了。

5,这个时候n-1个盘子在mid处,也就是在B那里。我们是不是可以这样想,也是相当于移动n-1个盘子,只不过换成了从B到C。

6,所以这里是不是直接就是

move(mid,from,to,level-1);

7,但是最后必须要考虑一个问题,也就是最后一个盘子的时候怎么移动,不就是直接移动吗,判断条件是level==1,直接

if(){System.out.println(from+"->"+to);}                  这样就完成了

 

最后总结一下!

  1. 一个问题的实现若是想用递归,那么它必然是有某种规律存在的,就是每个步骤必然有相似或者说循环的地方。
  2. 大问题用不了递归,不然会栈溢出,或者说递了但是归不了,所以是解决小问题的。
  3. 递归关键点两个,第一是抽取公共点,可能是有某个公式。第二是要找到归的条件,也就是判断条件,这个条件一般会出现在头或者尾。
  4. 递归其实也不好用啦,也就是简化代码量,比较灵活,能用就用,能不用就尽量不用。

 

 

 

 

 

 

这篇关于递归汉诺塔《Blind-Stab》的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/234351

相关文章

PHP实现二叉树遍历(非递归方式,栈模拟实现)

二叉树定义是这样的:一棵非空的二叉树由根结点及左、右子树这三个基本部分组成,根据节点的访问位置不同有三种遍历方式: ① NLR:前序遍历(PreorderTraversal亦称(先序遍历)) ——访问结点的操作发生在遍历其左右子树之前。 ② LNR:中序遍历(InorderTraversal) ——访问结点的操作发生在遍历其左右子树之中(间)。 ③ LRN:后序遍历(PostorderT

oracle11.2g递归查询(树形结构查询)

转自: 一 二 简单语法介绍 一、树型表结构:节点ID 上级ID 节点名称二、公式: select 节点ID,节点名称,levelfrom 表connect by prior 节点ID=上级节点IDstart with 上级节点ID=节点值 oracle官网解说 开发人员:SQL 递归: 在 Oracle Database 11g 第 2 版中查询层次结构数据的快速

Leetcode面试经典150题-128.最长连续序列-递归版本另解

之前写过一篇这个题的,但是可能代码比较复杂,这回来个简洁版的,这个是递归版本 可以看看之前的版本,两个版本面试用哪个都保过 解法都在代码里,不懂就留言或者私信 class Solution {/**对于之前的解法,我现在提供一共更优的解,但是这种可能会比较难懂一些(思想方面)代码其实是很简洁的,总体思想如下:不需要排序直接把所有数放入map,map的key是当前数字,value是当前数开始的

【UVA】10651-Pebble Solitaire(直接递归或者记忆化)

不知道这个题UVA的数据是怎么的,用2个方法交了,第一次直接递归,第二次记忆化剪枝,时间竟然一样!? 直接郁闷了,简单的二进制表示状态和二进制运算。 14145176 10651 Pebble Solitaire Accepted C++ 0.009 2014-09-04 09:18:21 #include<cstdio>#include<algorithm>#inclu

笔试强训,[NOIP2002普及组]过河卒牛客.游游的水果大礼包牛客.买卖股票的最好时机(二)二叉树非递归前序遍历

目录 [NOIP2002普及组]过河卒 牛客.游游的水果大礼包 牛客.买卖股票的最好时机(二) 二叉树非递归前序遍历 [NOIP2002普及组]过河卒 题里面给的提示很有用,那个马的关系,后面就注意,dp需要作为long的类型。 import java.util.Scanner;// 注意类名必须为 Main, 不要有任何 package xxx 信息publ

HCIA--实验十:路由的递归特性

递归路由的理解 一、实验内容 1.需求/要求: 使用4台路由器,在AR1和AR4上分别配置一个LOOPBACK接口,根据路由的递归特性,写一系列的静态路由实现让1.1.1.1和4.4.4.4的双向通信。 二、实验过程 1.拓扑图: 2.步骤: (下列命令行可以直接复制在ensp) 1.如拓扑图所示,配置各路由器的基本信息: 各接口的ip地址及子网掩码,给AR1和AR4分别配置

HDU 2064 汉诺塔III(水题)

题目: http://acm.hdu.edu.cn/showproblem.php?pid=2064 题目大意: 有三根杆,求把n个圆盘从左边移到右边,最少需要移动圆盘的次数。移动规则为大盘不能放在小盘上,比原始的汉诺塔题改变的地方是,只能通过中间的杆往左右两边的杆移动。 心得: 此题心得在题外,不在题内,初看此题,尼玛吓了一跳,好像很难的样子,手贱百度了一下,只注意到俩字“水题”,赶紧

Winform中在窗体中的Paint事件中重绘会导致递归问题?

在 WinForms 应用程序中,如果在窗体的 Paint 事件处理程序中不断调用 Invalidate 方法,确实可能会导致递归调用的问题。这是因为每次调用 Invalidate 方法时,都会向消息队列添加一个绘制消息,当消息队列中的绘制消息被处理时,会触发 Paint 事件。如果 Paint 事件处理程序中又调用了 Invalidate,就会形成一个循环,导致递归调用 Paint 事件,这

每日OJ_牛客_求和(递归深搜)

目录 牛客_求和(递归深搜) 解析代码 牛客_求和(递归深搜) 求和_好未来笔试题_牛客网 解析代码         递归中每次累加一个新的数,如果累加和大于等于目标,结束递归。此时如果累加和正好等于目标,则打印组合。向上回退搜索其它组合。此题本身就是一个搜索的过程,找到所有的组合。 #include <iostream>#include <cmath>#in

归并排序-非递归实现

归并排序的非递归实现  我们可以把 一个数组 先拆分成 最小单元,这是分, 拆分成最小单元之后,我们对每个最小单元进行一次合并,这是治 最小单元 合并一次之后,我们继续 在上一次合并的基础上拆分,并且合并这是 合 , 直到 整个数组 只被拆成了两部分,在进行最终一次的和   我们画图举个例子 源代码如下,我们的merge的合并函数的参数是  原数组,左侧小数组索引位置开头,左侧小数