牛客练习赛12 B题

迷宫

时间限制:C/C++ 1秒,其他语言2秒
空间限制:C/C++ 262144K,其他语言524288K
64bit IO Format: %lld

题目描述
这是一个关于二维迷宫的题目。我们要从迷宫的起点 ‘S’ 走到终点 ‘E’,每一步我们只能选择上下左右四个方向中的一个前进一格。 ‘W’ 代表墙壁,是不能进入的位置,除了墙壁以外的地方都可以走。迷宫内的 ‘D’ 代表一道上锁的门,只有在持有钥匙的时候才能进入。而 ‘K’ 则代表了钥匙,只要进入这一格,就会自动地拿到钥匙。最后 ‘.’ 则是代表空无一物的地方,欢迎自在的游荡。

本题的迷宫中,起点、终点、门跟钥匙这四个特殊物件,每一个恰好会出现一次。而且,此迷宫的四周 (最上面的一行、最下面的一行、最左边的一列以及最右边的一列) 都会是墙壁。

请问,从起点到终点,最少要走几步呢?

输入描述:
输入的第一行有两个正整数H, W,分别代表迷宫的长跟宽。
接下来的H行代表迷宫,每行有一个长度恰为W的字串,此字串只包含'S', 'E', 'W', 'D ', 'K', '.'这几种字元。

输出描述:
请在一行中输出一个整数代表答案,如果无法从起点走到终点,请输出-1。

示例1
输入
4 12
WWWWWWWWWWWW
WE.W.S..W.KW
W..D..W….W
WWWWWWWWWWWW
输出
20
示例2
输入
6 6
WWWWWW
WEWS.W
W.WK.W
W.WD.W
W.W..W
WWWWWW
输出
-1

备注
4 ≤ H, W≤ 500
‘S’, ‘E’, ‘K’, ‘D’各出现恰好一次
迷宫的四周(最上面的一行、最下面的一行、最左边的一列以及最右边的一列) 都会是 ‘W’

问题简述

迷宫拿钥匙开门问题,从入口S去找E的路途中可能会遭到门阻隔,所以需要去找钥匙,再去开门再去E。

问题分析

要注意这道题的下面几种情况:①目标E没有被门阻隔,不需要找钥匙直接到达目标E。②去找钥匙开门(如果可以找到钥匙的话)再到E。(蒻蒻我一开始做的时候以为不找钥匙直接到E一定要比去找钥匙开门再到E路径短,后来经巨巨指导,要注意最后要比较一下这俩种路径哪个更短一些。我惯性思维的认为找钥匙需要走更多的路,但是不是这样的,有可能去找钥匙开门再到E可能要更短一些。)③找到钥匙开了门却依然到达不了E,输出-1。④目标E被门阻隔并且钥匙也找不到,输出-1。
如果可以到达E,最后输出情况①和情况②中路径较短的一个。

好,接下来是AC代码。

#include
#include
#include
#include
#define N 505
using namespace std;

char a[N][N];
int book[N][N], nxt[4][2] = {{1,0},{0,1},{-1,0},{0,-1}};
int n,m,stx,sty,endx,endy;

struct node{
    int x, y, step;
}note[N];

int judge(int x, int y)
{
    if(book[x][y]==1)
    return 0;
    if(x<0||x>=n||y<0||y>=m)
    return 0;
    if(a[x][y]=='W'||a[x][y]=='D')
    return 0;
    return 1;
}

int bfs()
{
    queue  q;
    node cur,next;
    cur.x = stx; cur.y =sty;
    cur.step = 0;
    book[stx][sty] = 1;
    q.push(cur);
    while(!q.empty())
    {
        cur = q.front();
        q.pop(); //队首出列
        if(cur.x==endx && cur.y==endy){    //找到目标
            return cur.step;
        }
        for(int i=0;i<4;i++){
            next.x = cur.x + nxt[i][0];
            next.y = cur.y + nxt[i][1];
            if( judge(next.x,next.y) == 0)   //判定
                continue;
            book[next.x][next.y] = 1;
            next.step = cur.step + 1;
            q.push(next);  //入列
        }
    }
    return 0;
}

int main()
{
    scanf("%d%d",&n,&m);
    int t1, t2, tx, ty, px, py;
    for(int i = 0; i < n; i++)
    {
        for(int j = 0; j < m; j++)
        {
            scanf(" %c",&a[i][j]);
            //记录坐标
            if(a[i][j]=='S'){ stx = i; sty = j;}
            if(a[i][j]=='K'){ tx = i; ty = j;}
            if(a[i][j]=='D'){ t1 = i; t2 = j;}
            if(a[i][j]=='E'){ px = i; py = j;}
        }
    }
    endx = px; endy = py;
    int first = bfs();       //首先看不找钥匙是否可以到达E
    memset(book,0,sizeof(book));
    endx = tx; endy = ty;
    int second = bfs();    //再去看找钥匙再去开门到E
    int nowStep = 0;
    if(first==0)   //如果不能直接到达E
    {
        if(second==0)   //如果找不到钥匙
        cout << "-1" << '\n';
        else{
            a[t1][t2] = '.'; //找到钥匙有门的地方就可以走了
            nowStep += second; //记录找钥匙所需要的步数
            stx = tx; sty = ty;
            endx = px; endy = py;
            memset(book,0,sizeof(book));
            int fina = bfs(); //再从钥匙的坐标再去E
            if(fina==0)
            cout << "-1" << endl;
            else
            cout << nowStep + fina << endl;
        }
    }else
    {
        a[t1][t2] = '.';
        nowStep += second;
        stx = tx; sty = ty;
        endx = px; endy = py;
        memset(book,0,sizeof(book));
        nowStep += bfs();
        int ans = first>nowStep?nowStep:first; //注意要比较一下情况1和情况2
        cout << ans << '\n';
    }
    return 0;
}

你可能感兴趣的:(算法,ACM,ACM)