zoukankan      html  css  js  c++  java
  • poj3669 Meteor Shower

    题目http://poj.org/problem?id=3669

    在一个矩形区域,区域大小为0<=X<=300, 0<=Y<=300,有所谓流星坠落,给出M个流星坠落的时间和地点,问最少需要多少时间步能走到一个安全的格子上,从而不会遭受流星的撞击。每个流星坠落在某一格点后,其相邻四个格点会遭到破坏。

    样例输入

    4
    0 0 2
    2 1 2
    1 1 2
    0 3 5

    样例输出

    5

    思路

    • 假如起点(0,0)不会遭受到流星的攻击(也不会因为周围格子被攻击而延伸至它),那么也就不用移动就能保证安全。如果起点会受到攻击,不论是何时,我们都需要离开起点,那么哪里才是安全的终点呢?只要这个点上不会在有流星的攻击,这个点就是安全的终点,我们bfs出第一个安全的终点就是最少的时间步。
    • 根据输入我们可以标记每个受流星攻击的网格的时间,注意如果一个网格有多次被流星摧毁的时间,那么取最小的作为这个网格的摧毁时间,同时需要将这个网格上下左右四个网格也“以最近摧毁”原则更新被摧毁的时间。

    代码

    #include <iostream>
    #include <memory.h>
    #include <queue>
    #include <utility>
    using namespace std;
    typedef pair<int, int> PII;
    
    const int N = 302;
    int dx[] = {0, 1, 0, -1};
    int dy[] = {-1, 0, 1, 0};
    int time[N][N];
    int g[N][N];
    int bfs(){
        time[0][0] = 0;
        if(time[0][0] >=  g[0][0]) return -1;   //如果起点在时间0就被炸了, 那么直接失败...
        if(g[0][0] == 0x3f3f3f3f) return 0;     //如果起点不会被炸, 那么就不用走了...
        queue<PII> q;                           
        q.push(make_pair(0, 0));
        int time_step = 0;
        while(!q.empty()){
            PII p = q.front(); q.pop();
            for(int i = 0; i < 4; ++i){
                int x = p.first + dx[i], y = p.second + dy[i];
                if(x >= 0 && x <= N && y >= 0 && y <= N){
                    if(time[x][y] == -1){    //这个网格没有被访问过(不能走回头路)
                        time[x][y] = time[p.first][p.second] + 1;
                        if(g[x][y] == 0x3f3f3f3f){    //到达"终点"
                            return time[x][y];
                        }
                        if(time[x][y] < g[x][y] && g[x][y] != 0x3f3f3f3f) q.push(make_pair(x, y)); //如果这个网格要被炸但目前是安全的, 那么放入队列中
                    }
                }
            }
        }
        return -1;
    }
    
    int main(){
        freopen("input.txt", "r", stdin);
        freopen("output.txt", "w", stdout);
        int M; scanf("%d", &M);
        memset(g, 0x3f, sizeof(g));      //g记录网格被摧毁的时间T, 初始为正无穷(0x3f3f3f3f), 表示不会被摧毁
        memset(time, -1, sizeof(time));  //time记录到某一网格的时间步数,-1表示未访问
        while(M--){
            int x, y, t;
            scanf("%d%d%d", &x, &y, &t);
            if(g[x][y] > t) g[x][y] = t;
            for(int i = 0; i < 4; ++i){     //按"最近时间摧毁"原则记录每个网格被摧毁的时间
                int nx = x + dx[i], ny = y + dy[i];
                if(nx >= 0 && nx <= N && ny >= 0 && ny <= N){
                    if(g[nx][ny] > t)
                        g[nx][ny] = t;
                }
            }
        }
        printf("%d", bfs());
    }
    
  • 相关阅读:
    叩开抽象的大门(2)——依赖于抽象
    威老迷宫探险第二季如何更面向对象
    更佳的封装之路面向对象的封装思想
    威老的迷宫探险
    重用,我要重用!!!
    威老出国记,什么是引用,别名。
    叩开抽象的大门(1)——抽象类、接口
    maven常用命令
    大公司喜欢问的问题
    java 发送http请求
  • 原文地址:https://www.cnblogs.com/patrolli/p/12194415.html
Copyright © 2011-2022 走看看