hdu 5040 Instrusive【BFS+优先队列】

2014北京网络赛09题,hdu 5040


这次网络赛真是惨,也怪做题策略没想好,当时切完签到题之类的水题之后,马上就去看06青蛙那题去了。结果被那只死青蛙给坑惨了T_T。。。搞了四小时没搞出来...跪给那只青蛙了。。。本来当时是准备要做这道题的,题目描述也是好蛋疼,有人说这题不如直接去看Clarification,不看题目了,这也说明这题题目描述确实不清晰,虽然没这么夸张,题目还是得看的。


重新看这道题,不是很难,最短时间的话,那当然想到BFS,纯BFS的话,只是最短步数,所以我们需要+优先队列。状态的话,只需要坐标(x,y)还有时间cur_t%4这三个元素就行了。


解释一下为什么是cur_t%4,仔细想想会发现,这一题每个状态之间的不同点,除了当前位置的坐标,还有所有监视器的转向,而这个转向是以4为周期的没错吧,所以,只要某两个状态的坐标相同,且t%4也相同的话,那么这就是一个状态,压进队列一次就不用再压第二次了。除了这一点,其他应该没什么难点。后来写了一份代码,一炮就过了。。T_T为什么当时没先做这个?


代码:

#include <iostream>
#include <cstdio>
#include <queue>
#include <cstring>
#include <algorithm>
#define N 511
using namespace std;
int n,dir[4][2]={{0,-1},{-1,0},{0,1},{1,0}};//E==0,S==1,W==2,N==3
int change[110];
bool vis[N][N][5];
char a[N][N];
struct node
{
    int x,y,t;
    node(){;}   //构造函数
    node(int xx,int yy,int tt){x=xx,y=yy,t=tt;}//构造函数
    bool operator<(const node s)const{return t>s.t;}
};
int check(int x,int y,int t)//return 1代表该位置当前被灯照到,2代表当前位置没有灯照射,也可以走,0代表不可达
{
    if(x<1||x>n||y<1||y>n||a[x][y]=='#') return 0;
    if(change[a[x][y]]!=-1) return 1;
    for(int i=0;i<4;i++)//要知道当前点有没有被照到,需要查看当前点四个方向上有没有灯朝向这个点
    {
        int xx=x+dir[i][0],yy=y+dir[i][1];
        if(xx<1||xx>n||yy<1||yy>n) continue;
        int tmp=change[a[xx][yy]];//tmp>=0代表当前点有灯
        if(tmp>=0&&(tmp+t)%4==i) return 1;
        //(tmp+t)代表当前时间,%4代表此时灯的转向,稍微设计了一下方向函数dir,使得(tmp+t)==i的时候,说明该点被照到
    }
    return 2;
}
int bfs(int sx,int sy)
{
    memset(vis,0,sizeof(vis));
    priority_queue<node> que;
    que.push(node(sx,sy,0));
    while(!que.empty())
    {
        node tmp=que.top();que.pop();
        if(a[tmp.x][tmp.y]=='T') return tmp.t;
        int cur_t;
        //不移动的情况
        cur_t=tmp.t+1;
        if(!vis[tmp.x][tmp.y][cur_t%4]) vis[tmp.x][tmp.y][cur_t%4]=1,que.push(node(tmp.x,tmp.y,cur_t));
        int flag=check(tmp.x,tmp.y,tmp.t);//flag=1说明当前点被照亮
        for(int i=0;i<4;i++)
        {
            int cx=tmp.x+dir[i][0],cy=tmp.y+dir[i][1];
            int che=check(cx,cy,tmp.t);
            if(che==0) continue;
            if(che==1||flag==1) cur_t=tmp.t+3;//要移动的点或者当前点被照亮
            else if(che==2) cur_t=tmp.t+1;
            if(!vis[cx][cy][cur_t%4])
                vis[cx][cy][cur_t%4]=1,que.push(node(cx,cy,cur_t));
        }
    }
    return -1;
}
int main()
{
    #ifndef ONLINE_JUDGE
        freopen("D:/in.txt","r",stdin);
    #endif // ONLINE_JUDGE
    memset(change,-1,sizeof(change));
    change['E']=0,change['S']=1,change['W']=2,change['N']=3;
    int T,cas=1;
    scanf("%d",&T);
    while(T--)
    {
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
            scanf("%s",a[i]+1);
        int sx,sy;
        for(int i=1;i<=n;i++)
            for(int j=1;j<=n;j++)
            if(a[i][j]=='M') sx=i,sy=j;
        int ans=bfs(sx,sy);
        printf("Case #%d: %d\n",cas++,ans);
    }
    return 0;
}


你可能感兴趣的:(C++,搜索,ACM)