汉诺塔——这个世界确实是二进制的!

        在学习完递归后,今天就来简单说说经典的Hanoi(汉诺塔问题)。

        Hanoi出自一个印度的古老传说:在世界中心贝拿勒斯(在印度北部)的圣庙里,一块黄铜板上插着三根宝石针。印度教的主神梵天在创造世界的时候,在其中一根针上从下到上地穿好了由大到小的64片金片,这就是所谓的汉诺塔。不论白天黑夜,总有一个僧侣在按照下面的法则移动这些金片:一次只移动一片,不管在哪根针上,小片必须在大片上面。僧侣们预言,当所有的金片都从梵天穿好的那根针上移到另外一根针上时,世界就将在一声霹雳中消灭,而梵塔、庙宇和众生也都将同归于尽。看完这个传说想必一定很好奇这要移多少次世界才会毁灭吧。

        在理清这个传说后,可知这个传说中的要素有以下几点:1、3根柱子、64片金片;2、移动64片金片到另一柱子上;3、一次移动一片;4、小片必须在大片上面;5、求移动次数。在求解64片金片之前我们先移下3个金片。

        如图,A、B、C三根柱子,其中A柱自下而上依次叠放由大到小的三片圆片,3、2、1,将原片由A柱移至C柱,共计移动7次。(A柱:起始柱,B柱:过渡柱,C柱:目标柱)

        

        最简单粗暴的移动方式:将1、2全部移至过渡柱B,3移至目标柱C,再将过渡柱上的1、2移至目标柱C。那么其实不管如何圆片数量如何增加,最大的圆盘只会移动一次。问题的聚焦点就到了小圆片是如何移动的?3层汉诺塔的移动实际是求解2层汉诺塔的移动,那么再多层的汉诺塔都可以用这种方法解决。观察图片,不难发现1、2圆片移出用了3步,移回也用了3步。

        抽象为函数表示如下,通过函数表示很显然这是一个递归问题。

         知道这是一个递归问题后,那么就来尝试用代码描述一下这个过程吧。

#include <stdio.h>

int Hanoi(int, char, char, char);
//起始柱:pillar1;目标柱:pillar2;过渡柱:pillar3;
int Hanoi(int num, char pillar1, char pillar2, char pillar3)
{
	static int count = 0;
	if (num == 0)
	{
		return 0; 
	}
	else
	{
        Hanoi(num - 1, pillar1, pillar3, pillar2);//模拟小圆片移出
        count++;
		printf("第%d步,将 %d 号圆盘 %c —> %c \n", count, num, pillar1, pillar2);
		Hanoi(num - 1, pillar3, pillar2, pillar1);//模拟小圆片移动到目标柱
	}
	return 0;
}

int main()
{
	int num = 0;
	char pillar1 = 'A',
		 pillar2 = 'C',
		 pillar3 = 'B';	
	scanf("%d", &num);
	Hanoi(num, pillar1, pillar2, pillar3);


	return 0;
}

        运行结果如下:

      此外,汉诺塔问题与二进制的关系,在求解汉诺塔次数的时候能够很直观的感受到。  汉诺塔移动的过程可近似为二进制的计数。每一个圆片都代表着一个2进制位,可以根据下表可以很直观的看到,其实圆片个数就代表是要达到的二进制位数,而移动次数可以解释为当前还需要多少才可以进位。由此可以推导出公式2^n - 1

圆片数量/片次数(DEC)次数(BIN)
111
2311
37111
4151111
.........

        写到此,不禁感叹这个世界确实是二级制的世界啊!

        文末放上一条B站视频链接供学习参考:用二进制来解汉诺塔问题

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
汉诺塔问题是一个经典的问题,源于印度一个古老传说。问题是将一根柱子上的64片黄金圆盘按照大小顺序重新摆放到另一根柱子上,且在任何时候,小圆盘上不能放大圆盘,且一次只能移动一个圆盘。 汉诺塔问题的循环算法可以使用迭代的方式来解决。下面是一个示例代码: ```cpp #include <iostream> #include <stack> #include <string> using namespace std; void hanoiTowerIterative(int N, string from, string to, string help) { stack<int> s1, s2, s3; int moves = (1 << N) - 1; for (int i = N; i > 0; i--) { s1.push(i); } if (N % 2 == 0) { swap(s2, s3); } for (int i = 1; i <= moves; i++) { if (i % 3 == 1) { if (s1.empty() || (!s2.empty() && s2.top() < s1.top())) { cout << "Move " << s2.top() << " from " << to << " to " << from << endl; s1.push(s2.top()); s2.pop(); } else { cout << "Move " << s1.top() << " from " << from << " to " << to << endl; s2.push(s1.top()); s1.pop(); } } else if (i % 3 == 2) { if (s1.empty() || (!s3.empty() && s3.top() < s1.top())) { cout << "Move " << s3.top() << " from " << help << " to " << from << endl; s1.push(s3.top()); s3.pop(); } else { cout << "Move " << s1.top() << " from " << from << " to " << help << endl; s3.push(s1.top()); s1.pop(); } } else { if (s2.empty() || (!s3.empty() && s3.top() < s2.top())) { cout << "Move " << s3.top() << " from " << help << " to " << to << endl; s2.push(s3.top()); s3.pop(); } else { cout << "Move " << s2.top() << " from " << to << " to " << help << endl; s3.push(s2.top()); s2.pop(); } } } } int main() { hanoiTowerIterative(3, "A", "B", "C"); return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ZHOUZH_093

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值