[USACO2.4] 两只塔姆沃斯牛 The Tamworth Two
题目描述
两只牛逃跑到了森林里。Farmer John 开始用他的专家技术追捕这两头牛。你的任务是模拟他们的行为(牛和 John)。
追击在 $10 \times 10$ 的平面网格内进行。一个格子可以是:一个障碍物,两头牛(它们总在一起),或者 Farmer John。两头牛和 Farmer John 可以在同一个格子内(当他们相遇时),但是他们都不能进入有障碍的格子。
一个格子可以是:
`.` 空地;
`*` 障碍物;
`C` 两头牛;
`F` Farmer John。
这里有一个地图的例子:
plain
*...*.....
......*...
...*...*..
..........
...*.F....
*.....*...
...*......
..C......*
...*.*....
.*.*......
```
牛在地图里以固定的方式游荡。每分钟,它们可以向前移动或是转弯。如果前方无障碍(地图边沿也是障碍),它们会按照原来的方向前进一步。否则它们会用这一分钟顺时针转 90 度。 同时,它们不会离开地图。
Farmer John 深知牛的移动方法,他也这么移动。
每次(每分钟)Farmer John 和两头牛的移动是同时的。如果他们在移动的时候穿过对方,但是没有在同一格相遇,我们不认为他们相遇了。当他们在某分钟末在某格子相遇,那么追捕结束。
读入十行表示地图。每行都只包含 10 个字符,表示的含义和上面所说的相同。保证地图中只有一个 `F` 和一个 `C`。`F` 和 `C` 一开始不会处于同一个格子中。
计算 Farmer John 需要多少分钟来抓住他的牛,假设牛和 Farmer John 一开始的行动方向都是正北(即上)。 如果 John 和牛永远不会相遇,输出 0。
输入格式
输入共十行,每行 10 个字符,表示如上文描述的地图。
输出格式
输出一个数字,表示 John 需要多少时间才能抓住牛们。如果 John 无法抓住牛,则输出 0。
题解
分析
这道题简单来说就是,输入地图,一步一步走,适时转向,几步达到同一个点
首先题目给了初始位置,因此先通过输入的地图找到F和C的对应横纵坐标,再进行移动,我本想定义双指针去使它们达成一致,但准备步骤较为冗长,不如直接使用其横纵坐标进行标记,移动。
接下来是移动部分,由于每次都是顺时针旋转,因此一定会有如下对应关系:
北对东
东对南
南对西
西对北
以上便是顺时针的旋转方向
便于对上北下南左西右东的运动状态,我们不妨定义一个变量,同时为四个方向分别进行标记,用于明了每一次需要进行的操作是什么!
其次,两个物体的移动需要直接改写运动前后位置的内容吗?我的处理办法是“交换”!交换可占据的位置的内容,看起来或许就像一个动态图!
最后,转向只需判断下一位置是否越界或者为‘*’!
(这是我做题前的一个小草稿,可以参考一下!)
下见代码!
代码
#include<iostream>
#include<cstring>
using namespace std;
int i,j,a,b;
int t=0; //时间
char map[11][11]; //地图函数
int labelf=1; //F的操作指示
int labelc=1; //C的操作指示
void swap(char p,char q) //交换位置的内容的函数
{
p^=q;
q^=p;
p^=q;
}
void north(int *x,int *y,int *label) //向北方向移动的函数
{
if(map[*x-1][*y]=='*'||*x==1)
*label=4;
else
{
a=--*x;b=*y;
swap(map[a+1][*y],map[a][*y]);
}
}
void south(int *x,int *y,int *label) //向南方向移动的函数
{
if(map[*x+1][*y]=='*'||*x==10)
*label=3;
else
{
a=++*x;b=*y;
swap(map[a-1][*y],map[a][*y]);
}
}
void west(int *x,int *y,int *label) //向西方向移动的函数
{
if(map[*x][*y-1]=='*'||*y==1)
*label=1;
else
{
a=*x;b=--*y;
swap(map[a][b+1],map[a][b]);
}
}
void east(int *x,int *y,int *label) //向东方向移动的函数
{
if(map[*x][*y+1]=='*'||*y==10)
*label=2;
else
{
a=*x;b=++*y;
swap(map[a][b],map[a][b-1]);
}
}
void act(int *label,int *x,int *y) //每次进行的操作判断,以及分配操作
{
if(*label==1)
north(x,y,label);
else if(*label==2)
south(x,y,label);
else if(*label==3)
west (x,y,label);
else if(*label==4)
east (x,y,label);
}
int main()
{
int fx,fy,cx,cy; //F与C分别的横纵坐标
for(i=1;i<=10;i++)
for(j=1;j<=10;j++)
{
cin>>map[i][j];
if(map[i][j]=='F')
{
fx=i;fy=j;
}
if(map[i][j]=='C')
{
cx=i;cy=j;
}
}
while(fx!=cx||fy!=cy) //判断条件:有一个不等就不能退出循环
{
t++;
act(&labelf,&fx,&fy);
act(&labelc,&cx,&cy);
if(t==1000000) //一个足够大的范围,确保能找到答案
{
cout<<"0"<<endl;
return 0;
}
}
cout<<t<<endl;
return 0;
}
至此,此题结束!
路远常逢!
输入与输出
输入
*...*..... ......*... ...*...*.. .......... ...*.F.... *.....*... ...*...... ..C......* ...*.*.... .*.*......
输出
49