汉诺塔解析(图解)

ps:一时学不会也没关系,过一个月再自己试试说不定就学会了

ps:图片可能加载有点慢

题目:

三个柱子,标号为1,2,3

现在告诉你柱子1上套有n个盘,问你如何把全部盘从柱子1移到柱子3

注意:盘子顺序必须时刻保持从上到下是从小到大的,一次只能移一个盘

 

基本思路:

现在有3个柱子,分别标号为1,2,3,然后会输入一个n,代表初始的时候有几个盘子

特殊说明:输出过程的时候,把 1 -> 3 这样的结果看作从柱子1移最上面那个盘子到柱子3

比如说当三个盘子时

3
1 -> 3
1 -> 2
3 -> 2
1 -> 3
2 -> 1
2 -> 3
1 -> 3

 

然后开始循序渐进地解决这题

#1:

假如把一开始的n个盘子看成两部分,上面的小的和下面的大的,那么是不是只要三步就可以了

@1;把小的从1移到2

@2:把大的从1移到3

@3:把小的从2移到3

这里,1是起始柱子,2是中转站的暂时柱子,而3是终点柱子,理解这个下面的拆分就容易理解些

 

#2:

刚刚用的是绑定法,但是刚刚的三步,如果上面绑定的小的不止一个盘子,实际上就不能实现,因为上面的一群盘子不可能直接一次性移过去(一次只能移一个盘)

可是我们刚刚对于那一群小盘的操作是不是可以又拆分呢

也就是一开始对n-1个盘和1个盘进行了移动,那n-1个盘又可以拆分为n-2个盘和1个盘的移动

 

#3:具体的实现及代码

可以看到,就是把小的群移动到中转站,然后大的底盘移到终点,最后小的群移动到终点

其中,每一次小的群移动又是新的拆分,即以小的群中最下面那个做新的地盘,其他的又是更小更新的小的群

 

也许有人会纳闷为什么传入hanno的实参老是变化,比如说

hanno(temp,end,start,n-1);

这是因为hanno的意义就是以第一个参数为起点,以第二个参数为终点,以第三个参数为中转站,以第四个参数为当前还剩多少个盘的转移,刚刚那一步temp也许是做中转站,可是如果我们待会需要把temp的盘移到end那不仅是要把temp放第一个,end放第二个了吗,如此一来,start只能放第三个当这次的中转站

代码:

#include<bits/stdc++.h>
using namespace std;
int n;
/*
 * hanno函数的意义是把当前n个盘从start移到end,用temp作为中转站
 */
int hanno(int start,int end,int temp,int n)
{
    if(n-1>=1)//如果上面的部分起码还有一个,就还要拆 
        hanno(start,temp,end,n-1);//把小的群移到中转站,即从start移到temp,在这过程中end是充当临时存放的地方

    printf("%d -> %d\n",start,end);//把底盘从start移到end

    if(n-1>=1)//如果上面的部分起码还有一个,就还要拆 
        hanno(temp,end,start,n-1);//把小的群从中转站移到终点,即从temp移到end,也就是这时候start是充当临时存放的地方
}

int main()
{
    cin>>n;
    hanno(1,3,2,n);
}

 

posted @ 2018-12-01 11:15  ZYacmer  阅读(5021)  评论(0编辑  收藏  举报