0
点赞
收藏
分享

微信扫一扫

“汉诺塔”算法

问题描述:

      

“汉诺塔”问题有时大家有把它习惯的叫做“和尚搬塔”,它来自有古老的印度:在世界中心贝拿勒斯(在印度北部)的圣庙里,一块黄铜板上插着三根宝石针。印度教的主神梵天在创造世界的时候,在其中一根针上从下到上地穿好了由大到小的64片金片,这就是所谓的汉诺塔。不论白天黑夜,总有一个僧侣在按照下面的法则移动这些金片:一次只移动一片,不管在哪根针上,小片必须在大片上面。

问题分析:

             

            对于这个问题肯定是搬一辈子都搬不完的,以下分析其过程,看到底搬得完不?

             假如只有1个盘,则需搬动1次;

 

              假如只有3个盘,则需搬动8次;

              假如只有4个盘,则需搬动15次;

              ……

              不难归纳得到:当有n个盘时,需搬移2的n次方减1.

因此,当有64个盘(n=64)时,需要移动2的64次方减1次,这个数字是大的吓人吧,再转化一下就知道要多少年了,所以是一辈子也无法搬完的。

然而,我又为何要研究其算法啊?这个问题我还真不知道,但归咎我们还是可以思考一下其移动的过程,是如何移动的啊!

算法研究(实现):

           

         先将这个问题构建一个简单的数学模型,以方便研究。假设三根柱子分别编号为A,B,C,盘的个数为n,A柱子上从下到上按由大到小叠放着n个的圆盘,现在把所有盘子一个一个移动到柱子C上,并且每次移动同一根柱子上都不能出现大盘子在小盘子上方,请问至少需要多少次移动?

    肯定的是说把上面n-1个盘子移动到柱子B上,然后把最大的一块放在C上,实现第一轮的移动,最后把B上的所有盘子移动到C上,以此移动最终可以完成其要求。也就是个递归问题,由此得到类似阶乘算法的递归出表达式:   h(1) = 1   h(n) = 2*h(n-1)+1      (n>1)                    知道“斐波那契”数列的人,就比较容易推出这个表达式了。

其递归算法如下:(觉得这个算法是比较完美的,所以就用它了)

————————————————————————————————————-
#include”stdio.h”
void hanoi(int n,char one,char two,char three);
 void move(char x,char y);void main()
 {
         int m;
   printf(“————-hanoi—————-\n”);
         printf(“input the number of plates:”);
         scanf(“%d”,&m);
         printf(“The step to move %d plates:\n”,m);
         hanoi(m,’A',’C',’B');
 }
 //汉诺塔
 void hanoi(int n,char one,char two,char three)
 { 
         if (n==1)
         {
            move(one,three);        }
         else
         {
             hanoi(n-1,one,three,two);
             move(one,three);
             hanoi(n-1,two,one,three);
         }
 }
 //每次移动的顺序
 void move(char x,char y)
 {
         printf(“%c–>%c\n”,x,y);
 }

本算法是输出的移动盘的顺序!

     如果是6个盘,最终要全部从A移到C,即移动的顺序为:ABACBCABCACBAB

望诸位共同继续探讨“汉诺塔”问题,以求得到更完美的理解与算法。

举报

相关推荐

0 条评论