hdu-1026(bfs+优先队列)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1026

题意:输入n,m和一个n*m的矩阵,

.表示通路;

x表示墙;

n表示有一个怪物,消灭它需要n个时间。

求从(0,0)到(n-1,m-1)所需要的最短时间。

如果不存在,照题目格式输出;如果存在,将路径输出。

思路:广搜遍历求出最短路径,并用优先队列优化。

可以用递归逆向遍历出路径(开始没想到)。

 

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
struct Node{
    int x,y,step;
    friend bool operator<(Node a,Node b)
    {
        return b.step<a.step;
    }
};
char a[120][120];
int vis[120][120],fx[120][120],blood[120][120];
int n,m,tim,zz[4][2]={{0,1},{0,-1},{1,0},{-1,0}};
int bfs()
{
    priority_queue <Node> q;
    Node tmp;
    tmp.x=0;tmp.y=0;tmp.step=0;
    vis[0][0]=-1;
    q.push(tmp);
    while(!q.empty())
    {
        Node tmp=q.top();
        q.pop();
        if(tmp.x==n-1&&tmp.y==m-1) return tmp.step;
        for(int i=0;i<4;i++)
        {
            Node tp;
            tp.x=tmp.x+zz[i][0];
            tp.y=tmp.y+zz[i][1];
            if(tp.x<0||tp.x>=n||tp.y<0||tp.y>=m||vis[tp.x][tp.y]==-1) continue;
            tp.step=tmp.step+1+vis[tp.x][tp.y];
            vis[tp.x][tp.y]=-1;
            fx[tp.x][tp.y]=i+1;
            q.push(tp);
        }
    }
    return -1;
}
void Print(int x,int y)
{
    int tx,ty;
    if(fx[x][y]==0) return ;
    tx=x-zz[fx[x][y]-1][0];
    ty=y-zz[fx[x][y]-1][1];
    Print(tx,ty);
    cout<<tim++<<"s:("<<tx<<","<<ty<<")->("<<x<<","<<y<<")"<<endl;
    while(blood[x][y]--) cout<<tim++<<"s:FIGHT AT ("<<x<<","<<y<<")"<<endl;
}
int main(void)
{
    int i,j;
    while(cin>>n>>m)
    {
        memset(fx,0,sizeof(fx));
        memset(blood,0,sizeof(blood));
        for(i=0;i<n;i++) scanf("%s",a[i]);
        for(i=0;i<n;i++)
            for(j=0;j<m;j++)
            {
                if(a[i][j]=='.') vis[i][j]=0;
                else if(a[i][j]=='X') vis[i][j]=-1;
                else vis[i][j]=a[i][j]-'0',blood[i][j]=vis[i][j];
            }
        int ans=bfs();
        if(ans==-1)
        {
            cout<<"God please help our poor hero."<<endl;
        }
        else
        {
            cout<<"It takes "<<ans<<" seconds to reach the target position, let me show you the way."<<endl;
            tim=1;
            Print(n-1,m-1);
        }
        cout<<"FINISH"<<endl;
    }
    return 0;
}
View Code

 

posted @ 2018-10-19 15:52  麟阁  阅读(217)  评论(0编辑  收藏  举报