当前位置: 代码迷 >> 综合 >> [openjudge-搜索]哆啦A梦的时光机
  详细解决方案

[openjudge-搜索]哆啦A梦的时光机

热度:120   发布时间:2023-10-22 22:44:16.0

题目描述

描述

哆啦A梦有一个神奇的道具:时光机。坐着它,大雄和他的伙伴们能穿越时空,回到过去或者去到未来。
[openjudge-搜索]哆啦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);}
}