UVa 816 - Abbott's Revenge ( BFS )

题意

最多有9*9个节点的迷宫, 这个迷宫特殊的是, 每个节点可走的方向是特定的. 比如 1 2 WLF * 这个指令的意思是若朝向西来到点(1,2), 可以选择左转或直走
uva816

思路

BFS
一开始用数组做的模拟, 上交RE了, 不知道原因是什么
学习紫书写法, 利用队列和结构体

AC代码

#include <iostream>
#include <queue>
#include <cstdio>
#include <cstring>
#include <vector>

using namespace std;

char name[25], str[10];
int dir[4][2] = {{-1,0},{0,1},{1,0},{0,-1}};

//借助 strchr() 函数可以查找字符串s中首次出现字符c的位置这一功能将下列字母标序号
const char* ds = "NESW";  //标号为0-3 北东南西
const char* ts = "FLR";  //标号为0-2 直左右
int ds_id( char c ){ return strchr(ds, c) - ds; }
int ts_id( char c ){ return strchr(ts, c) - ts; }
int x1, y1, x2, y2, t;
char tt;

struct Node{
    int x, y, a;
    Node(int x = 0, int y = 0, int a = 0):x(x),y(y),a(a){}
};

bool judge[10][10][5][5], pla[10][10];
int vis[10][10][5];
Node pre[10][10][5];
void PRINT( Node u ){
    vector<Node> n;
    for(;;){
        n.push_back(u);
        if( vis[u.x][u.y][u.a] == 0 )   break;
        u = pre[u.x][u.y][u.a];
    }
    n.push_back(Node(x1,y1,t));
    int cnt = 0;
    for( int i = n.size()-1; i >= 0; i-- ){
        cnt++;
        if( cnt % 10 == 1 )     printf(" ");
        printf(" (%d,%d)",n[i].x, n[i].y);
        if( cnt % 10 == 0 || i == 0 )   putchar('\n');
    }
}
Node getv( Node u, int i ){  //i=0:直走,方向不变,  i=1:左转,  i=2:右转
    int a = u.a;
    if( i == 1 )   a = u.a-1 < 0 ? 3 : u.a-1;
    else if( i == 2 )   a = u.a+1 > 3 ? 0 : u.a+1;
    return Node(u.x+dir[a][0], u.y+dir[a][1], a );
}

void bfs()
{
    queue<Node> q;
    Node u(x1 + dir[t][0], y1 + dir[t][1], t);
    vis[u.x][u.y][u.a] = 0;
    q.push(u);
    while( !q.empty() ){
        Node u = q.front();
        q.pop();
        if( u.x == x2 && u.y == y2 ){
            PRINT(u);
            return;
        }
        for( int i = 0; i < 3; i++ ){
            Node v = getv(u, i);
            if( judge[u.x][u.y][u.a][i] && vis[v.x][v.y][v.a] < 0 && pla[v.x][v.y] ){
                vis[v.x][v.y][v.a] = vis[u.x][u.y][u.a] + 1;
                pre[v.x][v.y][v.a] = u;
                q.push(v);
            }
        }
    }
    printf("  No Solution Possible\n");
    return;
}

void readin()
{
    scanf("%d%d %c %d%d",&x1, &y1, &tt, &x2, &y2);
    pla[x1][y1] = true;
    pla[x2][y2] = true;
    t = ds_id(tt);
    int aa, bb;
    while( scanf("%d",&aa) && aa){
        scanf("%d",&bb);
        pla[aa][bb] = true;
        while( scanf("%s",str) && strcmp(str, "*") != 0 ){
            int len = strlen(str);
            for( int i = 1; i < len; i++ )
                judge[aa][bb][ds_id(str[0])][ts_id(str[i])] = true;
        }
    }
}

void solve()
{
    memset(judge, 0, sizeof(judge));
    memset(vis, -1, sizeof(vis));
    memset(pla, 0, sizeof(pla));
    readin();
    printf("%s\n",name);
    bfs();
}

int main()
{
    while( ~scanf("%s",name) && strcmp(name, "END") != 0 )
        solve();
    return 0;
}

RE代码

#include <iostream>
#include <cstdio>
#include <cstring>

using namespace std;

const int maxn = 10000;
char name[25], str[10];
int dir[4][2] = {{-1,0},{0,1},{1,0},{0,-1}};
int pre[maxn];
//借助 strchr() 函数可以查找字符串s中首次出现字符c的位置这一功能将下列字母标序号
const char* ds = "NESW";  //标号为0-3 北东南西
const char* ts = "FLR";  //标号为0-2 直左右
int ds_id( char c ){ return strchr(ds, c) - ds; }
int ts_id( char c ){ return strchr(ts, c) - ts; }
int x1, y1, x2, y2, t;
char tt;
bool flag;

struct Node{ int x, y, a; };
Node p[maxn];
bool judge[10][10][5][5], vis[10][10][5], pla[10][10];

int cnt = 0;
void PRINT( int a ){
    int t = pre[a];
    if( t != 0 )
        PRINT(t);
    else{
        printf(" (%d,%d)",x1, y1);
        printf(" (%d,%d)",p[0].x, p[0].y);
        cnt += 2;
    }
    printf(" (%d,%d)",p[a].x, p[a].y);
    cnt += 1;
    if( cnt % 10 == 0 ) putchar('\n');
}

void bfs()
{
    int head = 0, tail = 1, m, n, o, mm, nn, oo;
    p[0].x = x1 + dir[t][0], p[0].y = y1 + dir[t][1], p[0].a = t;
    pre[0] = -1;
    while( head < tail ){
        m = p[head].x, n = p[head].y, o = p[head].a;
        //cout << m << ' ' << n << ' ' << o << endl;
        if( m == x2 && n == y2 ){
            PRINT(head);
            flag = true;
            return;
        }
        for( int i = 0; i < 3; i++ ){
            if( judge[m][n][o][i] ){  //i=0:直走,方向不变,  i=1:左转,  i=2:右转
                if( i == 0 )    oo = o;
                else if( i == 1 )   oo = o-1 < 0 ? 3 : o-1;
                else if( i == 2 )   oo = o+1 > 3 ? 0 : o+1;
                mm = m + dir[oo][0];
                nn = n + dir[oo][1];
            }
            if( !vis[mm][nn][oo] && pla[mm][nn] ){
                //cout << mm << ' ' << nn << ' ' << endl;
                vis[mm][nn][oo] = true;
                p[tail].x = mm;
                p[tail].y = nn;
                p[tail].a = oo;
                pre[tail] = head;
                tail++;
            }
        }
        head++;
    }
    return;
}

void readin()
{
    scanf("%d%d %c %d%d",&x1, &y1, &tt, &x2, &y2);
    pla[x1][y1] = true;
    pla[x2][y2] = true;
    t = ds_id(tt);
    int aa, bb;
    while( scanf("%d",&aa) && aa){
        scanf("%d",&bb);
        pla[aa][bb] = true;
        while( scanf("%s",str) && strcmp(str, "*") != 0 ){
            int len = strlen(str);
            for( int i = 1; i < len; i++ )
                judge[aa][bb][ds_id(str[0])][ts_id(str[i])] = true;
        }
    }
}

void solve()
{
    memset(judge, 0, sizeof(judge));
    memset(vis, 0, sizeof(vis));
    memset(pla, 0, sizeof(pla));
    flag = false;
    readin();
    printf("%s\n",name);
    bfs();
    if( !flag ) printf(" No Solution Possible");
    putchar('\n');
}

int main()
{
    while( ~scanf("%s",name) && strcmp(name, "END") != 0 )
        solve();
    return 0;
}
posted @ 2018-03-06 19:19  JinxiSui  阅读(135)  评论(0编辑  收藏  举报