2022苏州市小学信息学奥赛T2-汉诺塔

本文探讨了汉诺塔问题,介绍了当有n个圆盘时,如何使用最少步数将其从A柱移动到C柱。针对具体步数x,给出了移动圆盘的编号及其移动路径。通过样例解析,揭示了数字和字母的规律,以解决大数据量下的汉诺塔问题。
摘要由CSDN通过智能技术生成

题目描述

汉诺塔问题是源于印度一个古老传说的益智玩具。大梵天创造世 界的时候做了三根金刚石柱子,在一根柱子上从上往下按照从小到小 大顺序摞着 64 片黄金圆盘。大梵天命令婆罗门把圆盘从上面开始按 从小到大顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能 放大圆盘,在三根柱子之间一次只能移动一个圆盘。 下图是圆盘个数 n=3 时的汉诺塔,这三根柱子从左向右依次编号为 A,B,C,开始 3 个圆盘都是在 A 柱上,从上往下依次编号为 1,2,3。 最后目标是把这三个圆盘都移动到 C 柱,我们知道最少移动步数是 7步。
Alt
第 1 步:1 号圆盘从 A 移动到 C

第 2 步:2 号圆盘从 A 移动到 B

第 3 步:1 号圆盘从 C 移动到 B

第 4 步:3 号圆盘从 A 移动到 C

第 5 步:1 号圆盘从 B 移动到 A

第 6 步:2 号圆盘从 B 移动到 C

第 7 步:1 号圆盘从 A 移动到 C 更一般的,如果是 n 个圆盘,最少移动步数是 2^n2
n
-1。 现在你的任务是:告诉你圆盘数 n(这些盘初始都在 A 柱上,而且从上往下编号 1 到 n),问你把这 n 个圆盘移动到 C 柱上,在最少步数移动过程中第 x 步移动的圆盘的编号,以及它从哪个柱子移动到哪个柱子?

输入

总共两行,第一行 1 个整数 n。 第二行 1 个整数 x。

输出

总共两行,第一行 1 个整数,表示第 x 步移动圆盘的编号。 第二行两个字母,中间有一个空格,表示第 x 步是从第一个字母的柱字移到第二个字母的柱子

样例

样例1
输入

3
6

输出

2
B C

样例2
输入

45 12345678910

输出

2
C A

提示
对于 20%的数据:1≤n≤16; 对于 50%的数据:1≤n≤32; 对于 100%的数据:1≤n≤64,1≤x≤2^n -1

题解

这道题先看数据量就知道肯定要用高精度,并且不能用普通方法,要找规律(至于为什么这样说后面我会说的)

朴素方法

#include<bits/stdc++.h>
using namespace std;
long long n,m,sum=0;
char a,b,c;
void f(int n,char a,char c,char b)
{
   
	if(n>0)
	{
   
	f(n-1,a,b,c);
	sum++;
	if(sum==m)
	{
   
	printf("%d\n",n);
	printf("%c %c",a,c);
    }
	f(n-1,b,c,a);
    }
}
int main()
{
   
	cin>>n;
	cin>>m;
	a='A';
	b='B';
	c='C';
	f(n,a,c,b);
}

long long的长度为2^32-1 跟题目的2^64-1相差甚远,所以内存就会炸。因为它是递归,又因为数据量太大所以还会超时

AC方法

既然要找规律,那么我们就先写一个打表的代码

#include<bits/stdc++.h>
#include<windows.h>
using namespace std;
long long n,sum=0;
char a,b,c;
void f(long long n,char a,char c,char b)
{
   
	if(n>0)
	{
   
	f(n-1,a,b,c);
	sum++;
	cout<<"第"<<sum<<"步"<<" "<<n<<"  "<<a<<c<<"    ";
	if(sum%3
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值