poj之旅——3669

题意描述与解析:

有个小文青去看流星雨,不料流星掉下来会砸毁上下左右中五个点。每个流星掉下的位置和时间都不同,求小文青能否活命,如果能活命,最短的逃跑时间是多少?

思路:对流星雨排序,然后将地图的每个点的值设为该点最早被炸毁的时间。如果起点一开始就被炸毁了的话,那小文青就直接挂花,否则bfs。

参考程序:

#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cstring>
#include<queue>
struct Meteor{
	int x,y,t;
};
typedef Meteor P;
bool visited[512][512];
int map[512][512],x[51002],y[51002],t[51002];
int N,last;
const int direction[5][2]={
	{1,0},
	{-1,0},
	{0,1},
	{0,-1},
	{0,0},
};
using namespace std;
int bfs(){
	memset(visited,0,sizeof(visited));
	queue<P> que;
	P current;
	current.x=0;
	current.y=0;
	current.t=0;
	que.push(current);
	while (que.size()){
		const P p=que.front();
		que.pop();
		for (int j=0;j<4;j++){
			current=p;
			current.x+=direction[j][0];
			current.y+=direction[j][1];
			current.t++;
			if (current.x >= 0 && current.y >= 0 && map[current.x][current.y] > current.t && !visited[current.x][current.y]){
				visited[current.x][current.y]=true;
				if (map[current.x][current.y]>last){
					return current.t;
				}
			    que.push(current);
			}
		}
	}
	return -1;
}
int main(){
	scanf("%d",&N);
	for (int i=0;i<N;i++)
		scanf("%d%d%d",&x[i],&y[i],&t[i]);
	memset(map,0x7F,sizeof(map));
	for (int i=0;i<N;i++){
		last=max(last,t[i]);
		for (int j=0;j<5;j++){
			int nx=x[i]+direction[j][0];
			int ny=y[i]+direction[j][1];
			if (nx>=0 && ny>=0)
				map[nx][ny]=min(t[i],map[nx][ny]);
		}
	}
	if (map[0][0]==0) printf("-1\n");
		else printf("%d\n",bfs());
	return 0;
}


你可能感兴趣的:(poj之旅——3669)