hdu 1983 Kaitou Kid - The Phantom Thief (2)

给了一个8*8的地图,时间要求却是5000ms ,似乎就是要遍历整个地图,枚举设置的封锁区域

一开始是想把所有可行的路径找出来,再看一下重复路径,判断应该设置多少个封锁区域,但是一旦设置了一个封锁区域,就又出现了不同的路径出来,这样,就跟枚举有何差别呢?

思路(大牛的):

封锁出口或者入口周围的格子. 
最多需要4个封锁点. 
所以我们可以采取这样的策略: 
1.寻找一条盗贼的可行路线,如果没有,返回0. 
2.计算封锁出口和入口四周需要的封锁点数量,取小的一个,假设是k,k <=4 
3.从少到多,遍历所有封锁点个数小于k的方案,验证是否是一条有效的覆盖方案
(可以通过是否阻止了1中的盗贼线路进行快速验证). 
如果有有效覆盖方案,返回这个方案的覆盖点值,否则继续. 
4.如果没有比k小的覆盖方案,返回k. 
首先,遍历的话,自然是想到了DFS,再遍历过程中要判断设置是否成功,则应该用较快的BFS了
不过没想到真的可以这样保存路径,我以为内存一定会爆呢,一直都不敢试一下,不过也许是对于这道题目吧,8*8的矩阵,及时清空队列的话,似乎也并不大
这是按照大牛的思路,不过是我的风格的代码,也许巧合,不过比大牛的快 了一点点哦
#include<iostream>
#include<queue>
using namespace std;
char map[10][10];
int n,m,T,ans,dir[4][2]={{0,1},{0,-1},{1,0},{-1,0}};
bool vis[10][10][2];//三维数组,第三维表示该点是否已拿到过宝石
struct node
{
	int x,y,num,step;
	int rox[64],roy[64];
	//x,y表示坐标,num表示是否拿到宝石,step表示步数或者时间,rox[]和roy[]分别保存到达该点的路径
};
node f;	//起始点
queue<node> Q;
void dfs(int deep)
{
	if(deep>ans) return ;//总共最多只需封锁四个区域,即入口或出口的四个方向
	node t;
	while(!Q.empty())//清空队列
		Q.pop();
	Q.push(f);
	memset(vis,0,sizeof(vis));
	vis[f.x][f.y][0]=1;
	int minstep=-1;
	while(!Q.empty())
	{
		 t=Q.front();
		 Q.pop();
		 node temp;
		if(map[t.x][t.y]=='E'&&t.num)
			{
			    minstep=t.step;
				break;
			}
		for(int k=0;k<4;k++)
		{
			int i=t.x+dir[k][0];
			int j=t.y+dir[k][1];
			if(i>n||i<1||j>m||j<1||map[i][j]=='#'||t.step>=T) continue;
			if(map[i][j]=='J') temp.num=1;
			else temp.num=t.num;
			if(vis[i][j][temp.num]) continue;
			for(int l=1;l<=t.step;l++)
			{
				temp.rox[l]=t.rox[l];
				temp.roy[l]=t.roy[l];
			}//保存路径
			vis[i][j][temp.num]=1;
			temp.x=i;temp.y=j;
			temp.step=t.step+1;
			temp.rox[temp.step]=i;
			temp.roy[temp.step]=j;
			Q.push(temp);
		}
	}
	if(minstep==-1) //minstep==-1表示该封锁区域设置成功,kid无法完成任务
	{
			if(deep<ans)
				ans=deep;
			return ;
	}
	for(int i=1;i<t.step;i++)
	{
		char cc=map[t.rox[i]][t.roy[i]];//保存原先的地图
		if(cc=='E'||cc=='S')//入口或出口不能封锁
			continue;
		map[t.rox[i]][t.roy[i]]='#';
		dfs(deep+1);//设置一个封锁区域后,继续遍历
		map[t.rox[i]][t.roy[i]]=cc;//将地图还原
	}
}
int main()
{
	int cas;
	cin>>cas;
	while(cas--)
	{
		cin>>n>>m>>T;
		for(int i=1;i<=n;i++)
			for(int j=1;j<=m;j++)
			{
				cin>>map[i][j];
				if(map[i][j]=='S')
					f.x=i,f.y=j;
			}
			f.num=0;f.step=0;
			ans=4;
			dfs(0);
			cout<<ans<<endl;
	}
	return 0;
}


 
posted @ 2011-08-11 17:04  枕边梦  阅读(462)  评论(0编辑  收藏  举报