题目描述
描述
哆啦A梦有一个神奇的道具:时光机。坐着它,大雄和他的伙伴们能穿越时空,回到过去或者去到未来。
有一天,大雄和他的伙伴们想穿越时空进行探险,可是时光机却出了一点故障,只能进行有限的时空穿越操作。大雄他们需要从现在出发,到达一个目标时间点进行探险,结束后再返回到现在,他们希望尽可能减少时光机的操作次数,你能帮助他们吗?
假设大雄和他的伙伴们出发的时间点(现在)为S(0 < S < 1,000,000),希望到达的时间点(目标)为T(0 < T < 1,000,000),已知时光机可以进行如下的时空穿越操作(X为正整数):
可以从任意时刻X穿越到X+1或者X-1时刻
可以从任意时刻X穿越到X*2时刻
当X为偶数时,可以从X时刻穿越到X/2时刻
请问,大雄和他的伙伴们从S时刻出发,先到达T时刻,再回到S时刻最少需要多少次时空穿越操作?
输入
输入的第一个数是一个正整数N,表示测试数据一共有N组(0 < N < 20)。
之后有N行,每一行包含两个正整数S和T,表示出发和到达时间点。S≠T
输出
输出包括N行,每一行一个正整数,表示每组测试数据对应的最少时光机操作次数。
样例输入
2
5 17
4 8
样例输出
8
2
提示
对于S=5,T=17:
操作如下:5->4->8->16->17->16->8->4->5
对于S=4,T=8:
操作如下:4->8->4
题目分析
这是一道十分类似BFS的走迷宫的题,给一个数X就可以使用一步的时间分支出X-1,X+1,X*2,X/2(当X%2==0时)四或三种走法。依次枚举得到最小步骤再*2即可。
代码实现
//之前写复杂了可以批评指正
#include<iostream>
#include<cstring>
using namespace std;
int S,head=0,tail=1,a[1000005],q,cont[1000005],c=0,x[3]={
1,-1};
bool b[1000005];
void print(int d)
{if(cont[d]!=0)print(cont[d]);//printf("%d\n",a[d]);c++;
}
void dfs(int n,int k)
{memset(b,0,sizeof(b));memset(a,0,sizeof(a));memset(cont,0,sizeof(cont));a[1]=n;b[n]=1;head=0;tail=1;while(head!=tail){head++;x[2]=a[head];for(int i=0;i<=3;i++){if(i==3){if(a[head]%2==0)q=a[head]/2;else break;}else q=a[head]+x[i];if(!b[q]&&q<=1000000&&q>=0){ tail++;a[tail]=q;cont[tail]=head;b[q]=1; if(a[tail]==k){print(tail);c--;return ;}}}}
}
int main()
{int n,k;scanf("%d",&S);for(int j=1;j<=S;j++){c=0;scanf("%d%d",&n,&k);dfs(n,k);dfs(k,n);printf("%d\n",c);}
}