hdu1072 BFS初体验

Nightmare

这道题目,本来我以为就是搜索题嘛,就直接写起来了DFS,但是试过几次之后发现WA个不停。然后我终于明白了DFS和BFS的区别。在这道题目的讨论区里都是用BFS写的,我就试着写了个BFS。结果直接AC了。。通过这个我感觉大致是这么个情况:当搜索最短路径的时候最好要选用BFS,当检查是否能走出去的时候选用DFS。用DFS求最短路径貌似不是每次都是最优解。它只保证能找到。当然了,这是我的愚见了。
这题的要点:
  • 每一个能走的点都能都很多次,而不是只能走一次。
  • 标记为4的点,是重置装置,虽然题目说该重置装置可以被使用多次,但这完全是误导罢了,如果你真的多次使用的话,那么你的程序一定会陷入死循环的。你说是不?所以,在你访问过一遍之后再去访问第二遍是没有任何意义的。因为接下来你要经历的状态,前面一定一定经历过喽,再走就会死循环,所以做个标记,走过就不要再走了。
这里我没有定义一个类来保存坐标。我是使用了一个简单的数学方法,我声明的一个对象是 queue<int> q。那么我是怎么用一个整型来保存二维坐标的呢?实际很简单
i*10+j 因为i,j都是个位数,所以很容易还原哦。
#include <iostream>
#include <queue>
using namespace std;
int n,m,si,sj,mini;
int ma[9][9];
int rest[9][9];
int step[9][9];
int dir[4][2]={0,1,0,-1,-1,0,1,0};
queue<int> q;
void bfs(int i,int j)
{
	q.push(i*10+j);
	while(!q.empty())
	{
		int front=q.front();
		q.pop();
		i = front/10;
		j = front%10;
		if(!rest[i][j])
			continue ;
		if(ma[i][j]==3)
		{
			if(mini==-1||mini>step[i][j])
				mini=step[i][j];
			continue ;
		}
		else if(ma[i][j]==4)
		{
			rest[i][j]=6;
			ma[i][j]=0;
		}
		for(int k=0;k<4;k++)
		{
			int x=i+dir[k][0];
			int y=j+dir[k][1];
			if(x<0||x>=n||y<0||y>=m||!ma[x][y]||rest[x][y]>=rest[i][j])
				continue;
			rest[x][y] = rest[i][j]-1;
			step[x][y] = step[i][j]+1;
			q.push(x*10+y);
		}
	}
	return ;
}
int main()
{//freopen("in.txt","r",stdin);
	int t;
	cin>>t;
	while(t--)
	{
		mini=-1;
		cin>>n>>m;
		memset(ma,0,sizeof ma);
		memset(rest,0 ,sizeof rest);
		memset(step,0 ,sizeof step);
		for(int i = 0;i<n;i++)
			for(int j=0;j<m;j++)
			{
				cin>>ma[i][j];
				if(ma[i][j]==2)
				{
					si=i;
					sj=j;
				}
			}
			ma[si][sj] = 0;
			rest[si][sj] = 6;
			bfs(si,sj);
			cout<<mini<<endl;
	}
	return 0;
}


你可能感兴趣的:(ACM,DFS,bfs,hdu1075)