汉诺塔问题详解及扩展(c++)

news2024/9/22 19:36:31

汉诺塔(Hanoi Tower)问题是一个著名的数学问题,它涉及到递归算法。问题的背景来源于一个传说:在印度的一个寺庙里,有三根金刚石柱和64个直径大小不一的金盘。僧侣们被命令将这些金盘从一根柱子按照从小到大的顺序移动到另一根柱子上,但必须遵守以下规则:

  1. 每次只能移动一个金盘。
  2. 每个金盘只能放在柱子上,不能放在其他地方。
  3. 任何时候,大盘不能放在小盘上面

假设现在我们有3根柱子,第一根上从下往上放着3,2,1三个不同大小的盘子 ,它大体的移动步骤是这样的:

首先我们先看要移动的1,2圆盘,目光聚集在1,2圆盘,3圆盘暂时不需要移动;

1:先将a桩上1,2圆盘中1圆盘移动到空柱c;

2:将剩下2圆盘移动到空柱b;

3:再将c柱上1圆盘移动到b柱上;

这是1,2圆盘具体的移动过程;

 

之后 我们审视全局,目光聚集在全部的圆盘上,将1,2圆盘看作一个整体,暂时不考虑它的具体移动过程,(具体移动过程就在上面)

1:先将a柱上1,2圆盘放在空柱c上,

2:再将3圆盘放在空柱b上;

3:再将c上1,2圆盘放在b上的3上

 

对比以下上面的图,有没有发现两个移动过程是不是几乎一样,可以看作第二图中嵌套着第一个图的过程

如果有n个盘子,那么我们要做的是先将n-1个盘子移动到c,然后将最后一个盘子也就是第n个盘子移动到b,再把c上的n-1个盘子移动到b其中n-1个盘子的移动过程就是将n-2个盘子移动到c,然后将第n-1个盘子移动到b 再将c柱上n-2个盘子移动到b,然后其中第n-2个盘子的移动过程是先将n-3个盘子移动到。。。。。

可以发现这整个其实就是一层套一层,其中的移动过程是重复的,因此我们可以用递归来解决;

代码:

#include<iostream>

#include<stack>
using namespace std;
stack<int>s[3];//每个盘子是先进后出,类似于栈,因此我们建立三个栈,相当于三根柱子;

void move(int a, int b) {//移动函数,将a柱子上的一个盘子移动到b柱子上
    int tmp = s[a].top();//取出a柱子上的栈顶元素,存在tmp临时变量中
    s[a].pop();//a柱子取出该元素,类似于把a柱子顶部的一个盘子拿走
    s[b].push(tmp);//将tmp存入b柱中,类似于把盘子放在b的顶部
    cout << a << "-->" << b<<endl;//输出这一步的移动过程
}
void hnt(int a, int b, int c, int n) {//汉诺塔函数的四个参数依次是//起始柱子经过柱子目标柱子每次移动的盘子数目
    if (n == 1) {//如果移动的盘子数为1了,即最后移动的盘子了,就把a中最后剩下的一个盘子移动到c柱子上
        move(a, c);
        return;
    }
    hnt(a, c, b, n - 1);//先将a柱子上的n-1个盘子经过c柱子移动到b柱子上
    move(a, c);//将a中剩下的一个盘子移动到c柱子上
    hnt(b, a, c, n - 1);//再将b柱子上n-1个盘子经过a移动c柱子上
}
int main() {
    int n;
    cin >> n;//输入盘子数n
    for (int i = n; i >= 1; i--)s[0].push(i);//先初始化0柱子,从大盘子到小盘子开始入栈
    hnt(0, 1, 2, n);//将n个圆盘从0柱子上经过1柱子移动到2柱子
    while (!s[2].empty()) {//输出最后2柱子上的元素,即2柱子上的盘子
        cout << s[2].top();
        s[2].pop();
    }
    return 0;
}

 尝试结果:

可以看到3个盘子时具体移动过程。


 

扩展:我们怎么求得整个过程移动的步数呢?3个盘子,我们可以通过数出来,如果是10,100个盘子呢?当然你也可以在函数move中增加一个变量记录移动的步数,但是, 这是递归,随着递归层次的不断增大,会导致栈溢出的问题,我在这里试了一下100个盘子,结果是运行了数十秒也得不到结果,一直在跑

因此我们要找另外的方法去计算这个步数;

通过模拟整个过程,我们可以发现

每一次移动的盘子数目其实是上一次移动盘子的数目乘2再加1

f(1)=1;

f(n)=f(n-1)+1+f(n-1)//类似于a柱子上n个盘子的步数等于将n-1个盘子移动到c柱的步数,加上a上最后一个盘子移动到b这一步,再加上c柱子上n-1个盘子移动到b柱子上的步数

因此我们可以通过循环写出代码计算步数

int main() {
    long long f[60];
    int n;
    cin >> n;
    f[1] = 1;
    for (int i = 2; i <= n; i++) {
        f[i] = 2 * f[i-1] + 1;
    }
    printf("%lld", f[n]);

现在我们可以试一下100盘子要移动多少步数

 

出错的原因呢是因为数据太大了,longlong都装不下了,longlong最大值是2^63-1大概20位数左右,我们试一下60

 可以看到有多少步吧。。。。60个盘子都已经差不多20位的步数了,怪不得之前跑100一直都计算不出结果;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2048443.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

北京移民服务机构亨瑞移民:汇集全球专业顾问 30年致力全球生活方式

01      北京移民服务机构亨瑞集团成立于1992年&#xff0c;作为一站式海外综合服务专业平台&#xff0c;30年来致力于为有意赴海外求学、拓展事业、居住、生活、工作的各行业人士提供全球教育规划、身份规划及资产全球配置解决方案。    北京移民服务机构亨瑞集团公司…

在 C#/.NET Core 的 Web API 中使用 Swagger 按模块和版本分组并实现排序

文章目录 前言步骤一&#xff1a;安装 Swashbuckle.AspNetCore步骤二&#xff1a;创建自定义特性步骤三&#xff1a;配置 Swagger 生成文档步骤四&#xff1a;标记控制器和方法总结 前言 在开发 RESTful API 时&#xff0c;良好的文档是必不可少的。Swagger 是一种广泛使用的 …

C/C++ 多线程[1]---线程创建+线程释放+实例

文章目录 前言1. 多线程创建2. 多线程释放3. 实例总结 前言 说来惭愧&#xff0c;写了很久的代码&#xff0c;一个单线程通全部。可能是接触的项目少吧&#xff0c;很多多线程的概念其实都知道&#xff0c;但是实战并没有用上。前段时间给公司软件做一个进度条&#xff0c;涉及…

【Docker系列】Docker 容器时区设置指南

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

中国式现代化产业水平数据集(2011-2021年)

中国式现代化是一种社会主义现代化模式&#xff0c;它具有自己独特的特征和发展路径。这一现代化模式以实现国家富强、民族振兴和人民幸福为目标&#xff0c;强调物质文明与精神文明的协调发展以及人与自然的和谐共生 本文将中国式现代化理念与现代化产业体系相结合&#xff0…

希亦、洁盟、苏泊尔眼镜清洗机哪款好用?热门眼镜清洗机测评总结

随着科学技术的发展&#xff0c;电子设备的升级&#xff0c;越来越多的人开始戴眼镜&#xff0c;而眼镜由于长时间的佩戴&#xff0c;镜框以及镜面都积累了一些灰尘以及人们肉眼所看不见的细菌&#xff0c;但是如果你使用普通的清洁方式去清洗的话肯定是清洗不干净的&#xff0…

记录|Label组件如何控制下边框为直线

目录 前言一、问题描述二、重绘三、效果展示更新时间 前言 参考文章&#xff1a; C# WinForm开发时&#xff0c;仅显示label的下边框怎么解决啊&#xff1f; 验证过&#xff0c;方法可靠。并增加控制绘制的直线粗细的功能。 一、问题描述 C# winform中想只给Label组件的下边框…

佳能FAX-L160G打印机驱动程序安装

佳能FAX-L160G打印机驱动程序安装笔记 1.访问佳能官方网站https://www.canon.com.cn或者相关驱动下载网站&#xff0c;搜索并下载适用于佳能FAX-L160G打印机的驱动程序。 在这里跳出来相关系列&#xff08;没有精确的型号&#xff09;&#xff0c;点进去搜索。 选择第一个驱动…

强化进度慢,武忠祥17堂课怎么听最高效?

武忠祥老师的17堂课怎么看&#xff1f; 我的建议是&#xff0c;别乱看&#xff01; 17讲内容包括15个主要专题和2个附加专题&#xff0c;整体课时量还是相当大的。 以2023版的课时长度为例&#xff0c;即使你全程以1.5倍速观看且不做任何停顿&#xff0c;平均每个专题至少也…

系统项目管理师----高级----前情调研

要成功通过信息系统项目管理师的考试&#xff0c;需要系统性的学习和多方面的准备。以下是详细的学习资源、推荐的书籍和具体的备考步骤。 一、学习资源与网站 1. 官方网站与指南 中国计算机技术职业资格网&#xff1a;这是软考的官方网站&#xff0c;提供考试大纲、政策法规…

Java编程 : 对象的本质

1.1 从机器视角到问题视角的演变 在计算机科学的发展历程中&#xff0c;我们见证了从机器视角到问题视角的深刻转变。这一转变不仅体现了编程语言和技术的进步&#xff0c;更反映了我们对问题解决方式理解的深化。 起初&#xff0c;计算机编程主要依赖于机器视角。汇编语言作…

FPGA串口调试中当电脑串口无法正常通信,设备管理器中“其它设备”位置显示“USB-Blaster”显示感叹号等问题应该怎么解决?

一、问题描述 当我们进行FPGA开发关于串口等试验的设计中&#xff0c;都需要用到串口&#xff0c;而要使用串口就需要先安装串口驱动&#xff0c;但是在安装驱动的过程中会出现各种各样的问题。 这里就出现了如图所示的停产问题&#xff1a; 在图片中我们可以看到FPGA要使…

【promise】Promise的几个关键问题 (三)

Ⅰ-如何改变 promise 的状态? (1) resolve(value): 如果当前是 pending 就会变为 resolved (2) reject(reason): 如果当前是 pending 就会变为 rejected (3) 抛出异常: 如果当前是 pending 就会变为 rejected Ⅱ-一个 promise 指定多个成功/失败回调函数, 都会调用吗? 当 pro…

Jenkins-拉取代码

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、Jenkins环境配置&#xff08;一&#xff09;配置Maven环境&#xff08;1&#xff09;Maven下载&#xff08;2&#xff09;将Maven上传服务器&#xff08;3&…

解决Win11点击睡眠后,唤醒系统,程序全部关闭的问题

网上只是给出修改“在此时间后关闭硬盘”的时间&#xff0c;结果点击【睡眠】&#xff0c;重新唤醒系统&#xff0c;程序依旧全部关闭。这是因为点击【睡眠】时&#xff0c;启动了【休眠】&#xff0c;Win11系统中&#xff0c;休眠再唤醒系统&#xff0c;会导致程序全部关闭。我…

如何在 3 分钟内免费在 AWS 上运行 RStudio

欢迎来到雲闪世界。谈到数据分析&#xff0c;我有理由从本地计算机迁移到云端。最突出的是&#xff0c;您可以运行无限数量的机器&#xff0c;而无需拥有或维护它们。此外&#xff0c;您可以在几分钟内根据需要扩大或缩小规模。如果您选择运行 t2.micro 服务器&#xff0c;您可…

ant design pro 如何去保存颜色

上图 就是实现这样的效果 后端是这样的&#xff0c;这个颜色肯定是存到字符串里的 这是第一步 import mongoose, { Schema, Document } from mongoose;interface IDiscountCard extends Document {title: string;subtitle: string;image: string;shopUrl: string;bgColor: s…

望繁信科技荣膺上海市浦东新区博士后创新实践基地称号

近日&#xff0c;上海望繁信科技有限公司&#xff08;简称“望繁信科技”&#xff09;凭借在大数据流程智能领域的卓越表现&#xff0c;成功入选上海市浦东新区博士后创新实践基地。这一荣誉不仅是对望繁信科技创新能力和技术实力的高度认可&#xff0c;也标志着公司在推动产学…

基于springboot养老院管理系统pf

TOC springboot332基于springboot养老院管理系统pf 第1章 绪论 1.1选题动因 当前的网络技术&#xff0c;软件技术等都具备成熟的理论基础&#xff0c;市场上也出现各种技术开发的软件&#xff0c;这些软件都被用于各个领域&#xff0c;包括生活和工作的领域。随着电脑和笔记…

Hive3:简单ETL实操案例

一、ETL概念简介 ETL&#xff1a; E&#xff0c;Extract&#xff0c;抽取 T&#xff0c;Transform&#xff0c;转换 L&#xff0c;Load&#xff0c;加载 从A抽取数据(E)&#xff0c;进行数据转换过滤(T)&#xff0c;将结果加载到B(L)&#xff0c;就是ETL 二、情景描述 聊天平…