hdu 1207 四柱汉诺塔

     递推,汉诺塔I的变形。

     这题真心没想到正确解法,越想越迷糊。这题看了别人题解过得,以后还是自己多想想,脚步太快并非好事。

     贴上分析:

   

  分析:设F[n]为所求的最小步数,显然,当n=1时,F[n]=1;当n=2时,F[n]=3;如同经典汉诺塔一样,我们将移完盘子的任务分为三步:
(1)将x(1<=x<=n)个盘从a柱依靠b,d柱移到c柱,这个过程需要的步数为F[x];
(2)将a柱上剩下的n-x个盘依靠b柱移到d柱(注:此时不能够依靠c柱,因为c柱上的所有盘都比a柱上的盘小)
     些时移动方式相当于是一个经典汉诺塔,即这个过程需要的步数为2^(n-x)-1(证明见再议汉诺塔一);
(3)将c柱上的x个盘依靠a,b柱移到d柱上,这个过程需要的步数为F[x];
第(3)步结束后任务完成。
故完成任务所需要的总的步数F[n]=F[x]+2^(n-x)-1+F[x]=2*F[x]+2^(n-x)-1;但这还没有达到要求,题目中要求的是求最少的步数,易知上式,随着x的不同取值,对于同一个n,也会得出不同的F[n]。即实际该问题的答案应该min{2*F[x]+2^(n-x)-1},其中0<x<n;
   AC代码:

#include<cstdio>
#include<algorithm>
using namespace std;
typedef unsigned long long LL; //重点注意无符号
const int maxn=65;
const int INF=1<<30;
LL ans[maxn];
LL power(LL a,LL n){ //快速幂
    LL w=1;
    while(n>0){
        if(n%2==1)
            w*=a;
        n/=2;
        a*=a;
    }
    return w;
}
void solve(){
    ans[0]=0;
    ans[1]=1;
    ans[2]=3;
    for(int i=3;i<=64;++i){
        ans[i]=INF;
        for(int j=1;j<i;++j)
            ans[i]=min(ans[i],ans[j]*2+power(2,i-j)-1);
    }
}
int main(){
    int n;
    solve();
    while(scanf("%d",&n)==1){
        printf("%lld\n",ans[n]);
    }
    return 0;
}


如有不当之处欢迎指出!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值