现在的位置: 首页 > 综合 > 正文

“汉诺塔”算法

2018年01月11日 ⁄ 综合 ⁄ 共 1661字 ⁄ 字号 评论关闭

 xcbeyond版权所有

http://www.xcbeyond.com/?p=669

问题描述:

      

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

问题分析:     

             

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

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

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

              假如只有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

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

【上篇】
【下篇】

抱歉!评论已关闭.