[USACO11OPEN]玉米田迷宫Corn Maze题解

题目链接

洛谷p1825
bzoj3299

以下描述针对于洛谷环境

这一题本来是普通的bfs,但坑点却很多,其中有一个疑似数据问题。

坑点1: 传送门可以多次走

bfs不扩张重复点,这是它比dfs快的原因之一。但这一题传送门可以多次走,比如这个样例:

5 5
#####
#.#.#
#A#A=
#.#@#
#####

传送门是强制传送的,没有选择,所以需要两次经过传送门,传过去再传回来。

解决方案:

就拿样例来说,从@到第四列的A会直接传到第二列的A,可以看做没经过第四列的A,而是直接到了第二列的A,稍微处理一下即可。

处理代码

//如果是传送门就直接跳到另一头
if(mp[xx][yy]>='A'&&mp[xx][yy]<='Z')
{
	int t=mp[xx][yy]-'A';
	//x0,y0,x1,y1分别为传送门两端的横纵坐标	
	xx=xx==x0[t]?x1[t]:x0[t];//跳转
	yy=yy==y0[t]?y1[t]:y0[t];
}

坑点2: 传送门不配全

疑似数据问题。就第四个点的传送门Z,其他都没问题。虽然洛谷没说,但其它oj都保证了传送门成对出现。

对于我的代码,它会跳转到第0行,第0列,而我判断能走的依据只有不为#,所以就越界了。我就直接把mp[0][0]赋为#,把单独的传送门看成#就行。

code:

#include
using namespace std;
const int dx[4]={1,0,-1,0},dy[4]={0,1,0,-1};
char mp[310][310];
int f[310][310],qx[90010],qy[90010],x0[26],y0[26],x1[26],y1[26],q1=1,q2;
int main()
{
	mp[0][0]='#';//把mp[0][0]赋为#,防止越界
	int n,m;
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++)
		for(int j=1;j<=m;j++)
		{
			cin>>mp[i][j];
			if(mp[i][j]=='@')//起点入队
			{
				qx[++q2]=i;
				qy[q2]=j;
				f[i][j]=1;
			}
		//预处理传送门两端的横纵坐标
		if(mp[i][j]>='A'&&mp[i][j]<='Z')
			{
				int t=mp[i][j]-'A';
				if(!x0[t])x0[t]=i,y0[t]=j;
				else x1[t]=i,y1[t]=j;
			}
		}
	while(q1<=q2)
	{
		int x=qx[q1],y=qy[q1];
		for(int i=0;i<4;i++)
		{
			int xx=x+dx[i],yy=y+dy[i];
            //如果是传送门就直接跳到另一头
			if(mp[xx][yy]>='A'&&mp[xx][yy]<='Z')
			{
				int t=mp[xx][yy]-'A';
				xx=xx==x0[t]?x1[t]:x0[t];
				yy=yy==y0[t]?y1[t]:y0[t];
			}
            //照常处理
            //用f同时代表步数和走没走过
			if(mp[xx][yy]!='#'&&!f[xx][yy])
			{
				f[xx][yy]=f[x][y]+1;
				qx[++q2]=xx;
				qy[q2]=yy;
				if(mp[xx][yy]=='=')
				{
                	//起点步数即为一,所以要减一
					printf("%d\n",f[xx][yy]-1);
					return 0;
				}
			}
		}
		q1++;
	}
	return 0;
}

你可能感兴趣的:(题解)