题目:
汉诺塔(又称河内塔)问题是源于印度一个古老传说的益智玩具。大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘。
现在蒜头君开始玩汉诺塔游戏,他放了n片黄金圆盘在第一根柱子上,从上到下依次编号为1−n, 1号圆盘最小,n号圆盘最大。蒜头君移动第i号圆盘的时候需要花费i点体力。现在蒜头君想把圆盘全部移动到底2根柱子上,移动过程中蒜头君必须准守游戏规则。
现在蒜头君想知道他完成游戏的最小移动次数和最少消耗的体力。
Input
输入一个正整数t(1<=t<=1000)代表数据的组数
之后t行每行输入一个正整数n(1≤n≤60)表示黄金圆盘的个数
Output
一行输出2个数,表示最小移动次数和最小消耗的体力,中间用一个空格隔开。共t行。
Sample Input 1
2
2
3
Sample Output 1
3 4
7 11
思路:
首先我想现就汉诺塔求移动次数的想法说一下,首先是递推思想,画个图说明一下便于理解:
之后
最后
由这个过程不难推导出递推公式:f(n) = f(n-1) + 1 + f(n-1) ,即 f(n) = 2 * f(n-1) + 1,通过这个递归式不难求出次数。不过还有更简单的,看了网友的题解我也长见识,有一个挺奥妙的规律在里面:
再来看看求体力的
这个题很经典,很不错,值得整理,上代码:
// 普通思路
#include"iostream"
using namespace std;
long long spe[61]; // 消耗的体力
long long han(int n){
if(n > 1){
return 2*han(n-1)+1;
}
if(n == 1){
return 1;
}
}
int main(){
int t;
cin >> t;
while(t--){
int n;
cin >> n;
long long pos = han(n);
spe[1] = 1;
for(int i = 2;i <= n;i ++){
spe[i] = 2*spe[i-1] + i;
}
cout << pos << " " << spe[n] << endl;
}
return 0;
}
// 规律化简版
#include<bits/stdc++.h>
using namespace std;
int main()
{
long long n,sum=1;
int t;
cin >> t;
while(t--){
cin>>n;
for(int i=1;i<=n;i++)
{
sum*=2;
}
cout<<sum-1<<" "<<(sum-1)*2-n<<endl;
sum = 1;
}
return 0;
}
脚踏实地,稳中求进,加油!!