ACWing 96 奇怪的汉诺塔 题解
xiaoh 2020-7-7
题意
现将汉诺塔的三柱换位四柱。对于的每一个求阶汉诺塔的最小操作步数。
题解
不妨记汉诺塔的柱子依次为,先回顾一下三柱汉诺塔的计算方法。我们先将个盘子从柱搬到柱,然后将号盘从柱搬到柱,再将个盘子从柱搬到。所以递推式为.
而四柱汉诺塔既然多了一根柱子,我们更不能把它闲着。我们可以将个盘子分成两部分,靠上面的一部分放到柱,此时我们是按照“四柱”来计算的,剩下的由于有一部分已经占了一根柱子,所以我们只能在“三柱”模式下将剩余的一部分盘子放到柱上,最后将号盘子放到柱上,最后再将两部分盘子分别从移动到上即可。所以递推式为。按照上述递推式递推即可。
Code
#include<bits/stdc++.h>
using namespace std;
template<typename T>
inline void read(T &x)
{
x=0;
int f=1;
char ch=getchar();
while(ch<'0'||ch>'9')
{
if(ch=='-') f=-1;
ch=getchar();
}
while(ch>='0'&&ch<='9') x=x*10+(ch^48),ch=getchar();
x*=f;
return;
}
template<typename T>
void write(T x)
{
if(x<0) putchar('-'),x=-x;
if(x>=10) write(x/10);
putchar(x%10+'0');
return;
}
const int MAXN=15;
const long long inf=999999999;
long long f[MAXN];
long long g[MAXN];
int main()
{
g[1]=1;
for(int i=2;i<=12;++i) g[i]=g[i-1]*2+1;
f[1]=1,f[2]=3;
for(int i=3;i<=12;++i)
{
f[i]=inf;
for(int j=1;j<i;++j) f[i]=min(f[i],2*f[j]+2*g[i-j-1]+1);
}
for(int i=1;i<=12;++i) cout<<f[i]<<endl;
return 0;
}