题意描述与解析:
有个小文青去看流星雨,不料流星掉下来会砸毁上下左右中五个点。每个流星掉下的位置和时间都不同,求小文青能否活命,如果能活命,最短的逃跑时间是多少?
思路:对流星雨排序,然后将地图的每个点的值设为该点最早被炸毁的时间。如果起点一开始就被炸毁了的话,那小文青就直接挂花,否则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; }