在学习完递归后,今天就来简单说说经典的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进制位,可以根据下表可以很直观的看到,其实圆片个数就代表是要达到的二进制位数,而移动次数可以解释为当前还需要多少才可以进位。由此可以推导出公式。
圆片数量/片 | 次数(DEC) | 次数(BIN) |
---|---|---|
1 | 1 | 1 |
2 | 3 | 11 |
3 | 7 | 111 |
4 | 15 | 1111 |
... | ... | ... |
写到此,不禁感叹这个世界确实是二级制的世界啊!
文末放上一条B站视频链接供学习参考:用二进制来解汉诺塔问题