Java 细致图解带你分析汉诺塔

 更新时间:2022年03月23日 11:21:30   作者:来自爪哇的bean  
汉诺塔问题是一个经典的问题。汉诺塔(Hanoi Tower),又称河内塔,源于印度一个古老传说。本文将用Java求解这一问题,感兴趣的可以学习一下

一、汉诺塔问题来源

汉诺塔(Tower of Hanoi),又称河内塔,是一个源于印度古老传说的益智玩具。大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘

二、问题分析

从简单问题开始

直接拿64个盘子来想,可能会比较难,我们可以先从1个盘子开始看,如下图:

一个盘子

A -> C 

只有一个盘子情况下,我们可以直接将 A 柱子上面的盘子移到 C 柱子上

需要移动一次

两个盘子

当有两个盘子时,我们也可以通过下面方式实现:

A -> B     A->C     B->C

需要移动3次

1.  A -> B

2.  A -> C

 3.  B -> C

 三个盘子

 当有三个盘子时,移动步骤如下:

A -> C     A -> B     C -> B     A -> C     B -> A     B -> C     A -> C

共需要移动7次 

 1.  A -> C

2.  A -> B

 3.  C -> B

4.  A -> C

 5.  B -> A

 6.  B -> C

 7.  A -> C

这就完成了3个盘子的移动

当有 4 个盘子时,这个问题其实就已经很复杂了

规律推导

1个盘子      移动1次

2个盘子      移动3次

3个盘子      移动7次

……

N 个盘子    移动 2^N - 1 次

那么64个盘子就是需要移动 2^64 - 1 次

三、解决问题

我们可以通过递归来解决这个问题,获得正确的移动方式

如果有N个盘子该怎么移动呢?

整体思路

我们可以先将 N - 1 个盘子从 A 柱借助 C 柱移动到 B 柱,再将 A 柱剩下的一个盘子移动到 C柱,然后将 B 柱上的 N - 1 个盘子借助 A 柱移动到 C 柱,就完成了所有柱子的移动(中间具体移动过程暂不讨论)

上代码

public static void hanoi(int num, String src, String help, String dest) {
    if (num == 1) {     // 只有一个盘子的时候直接移动
        System.out.print(src + "->" + dest + "  ");  // 将一个盘子从源柱子挪到目标柱子
    } else {
        hanoi(num - 1, src, dest, help);   // 将n - 1个盘子从源柱子借助目标柱子挪到辅助柱子
        System.out.print(src + "->" + dest + "  ");  // 将一个盘子从源柱子挪到目标柱子
        hanoi(num - 1, help, src, dest);  // 将辅助柱子上n - 1个盘子借助源柱子挪到目标柱子
    }
}
public static void main(String[] args) {
    hanoi(3, "A", "B", "C");
}

这段代码中 src 是源柱子,help是辅助柱子,dest 是目标柱子

这是一个二路递归

运行结果:

 这就成功完成了盘子的移动

四、婆罗门能否完成大梵天的任务

移动 64 个盘子需要多长时间

在这里我们假设婆罗门的人都非常聪明,不需要思考就直接能知道正确的移动方法,移动一个盘子需要一秒钟,一直不停的移

将2^64 - 1秒换算为年约为5849 4241 7355年(5849.42亿年),而地球存在至今不过45亿年,太阳系的预期寿命据说也就是数百亿年。真的过了5849.42亿年,不说太阳系和银河系,至少地球上的一切生命,连同梵塔、庙宇等,都早已经灰飞烟灭。

相关预言

有预言说,这件事完成时宇宙会在一瞬间闪电式毁灭。也有人相信婆罗门至今还在一刻不停地搬动着圆盘

计算机移动64个盘子需要多长时间 ?

我的电脑核心频率为2.90GHz,也就是每秒钟运算 29 亿次,那么移动 2^64 - 1次需要的时间约为201年

到此这篇关于Java 细致图解带你分析汉诺塔的文章就介绍到这了,更多相关Java 汉诺塔内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 通过java生成读取二维码详解

    通过java生成读取二维码详解

    这篇文章主要介绍了java二维码生成读取详解,二维码再生活在无处不在,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,下面和小编一起来学习一下吧
    2019-05-05
  • Java  匿名内部类详解及实例代码

    Java 匿名内部类详解及实例代码

    这篇文章主要介绍了Java 匿名内部类详解及实例代码的相关资料,需要的朋友可以参考下
    2017-02-02
  • IDEA取消SVN关联,再重新分享项目的操作

    IDEA取消SVN关联,再重新分享项目的操作

    这篇文章主要介绍了IDEA取消SVN关联,再重新分享项目的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-02-02
  • Java CAS与Atomic原子操作核心原理详解

    Java CAS与Atomic原子操作核心原理详解

    CAS(Compare and Swap)和Atomic原子操作是保证多线程并发安全的常用机制,能够高效地实现对共享变量的安全访问和修改,避免线程竞争导致的数据不一致和死锁等问题。它们的应用可以提高程序的并发性能和可维护性,是多线程编程中的重要工具
    2023-04-04
  • Spring自动注入失败的解决方法

    Spring自动注入失败的解决方法

    这篇文章主要介绍了Spring自动注入失败的解决方法,帮助大家更好的理解和学习使用Spring框架,感兴趣的朋友可以了解下
    2021-05-05
  • Spring Security 中细化权限粒度的方法

    Spring Security 中细化权限粒度的方法

    这篇文章主要介绍了Spring Security 中细化权限粒度的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • 一文带你深入理解Java AbstractQueuedSynchronizer

    一文带你深入理解Java AbstractQueuedSynchronizer

    在并发编程中,锁是一种保证线程安全的方式,这篇文章主要为大家介绍了AbstractQueuedSynchronizer(AQS)的数据结构及实现原理,感兴趣的小伙伴可以了解一下
    2023-07-07
  • 基于Column注解的columnDefinition用法

    基于Column注解的columnDefinition用法

    这篇文章主要介绍了Column注解的columnDefinition用法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • 常用输入字节流InputStream介绍

    常用输入字节流InputStream介绍

    下面小编就为大家带来一篇常用输入字节流InputStream介绍。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08
  • Java后端中dto、vo、entity的区别浅析

    Java后端中dto、vo、entity的区别浅析

    这篇文章主要给大家介绍了关于Java后端中dto、vo、entity区别的相关资料,文中通过实例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2023-01-01

最新评论