UVa (一道比较复杂的广搜) 816 Abbott’s Revenge

题意:

给出一个迷宫,在迷宫的节点处,面向某个方向只能向给定的方向转弯。给出起点和终点输出迷宫的最短路径,这里指的是刚刚离开起点的时刻,所以即使起点和终点重合路径也非空。

分析:

用三个变量来表示状态,r,c,dir,分别代表所处的位置和朝向。在输入数据的同时,也要初始化has_edge[r][c][dir][turn],代表处于(r, c, dir)这个状态时能否向turn转弯。

结构体数组p用来保存路径。

因为路径可能比较长,所以如果采用地轨输出的话,可能会栈溢出。代码中采用了动态数组来输出。

一直WA的原因:读入函数Input返回值是bool,而只在代码中写了return false; 却没有在最后加上 return true;

  1 //#define LOCAL
  2 #include <iostream>
  3 #include <cstdio>
  4 #include <cstring>
  5 #include <queue>
  6 #include <vector>
  7 using namespace std;
  8
  9 struct Node
 10 {
 11     int r, c;
 12     int dir;
 13     Node(int rr=0, int cc=0, int ddir=0):r(rr), c(cc), dir(ddir) {}
 14 };
 15
 16 const char* dirs = "NESW";
 17 const char* turns = "FLR";
 18 int dir_id(char c)    { return strchr(dirs, c) - dirs; }
 19 int turn_id(char c)    { return strchr(turns, c) - turns; }
 20 const int dr[] = {-1, 0, 1, 0};
 21 const int dc[] = {0, 1, 0, -1};
 22 int d[10][10][4];
 23 Node p[10][10][4];
 24 int r0, c0, r1, dir, c1, r2, c2;
 25 char name[100], c[100], s[100];
 26 bool has_edge[10][10][4][3];
 27
 28 bool inside(int r, int c)
 29 { return (r>=1 && r<=9 && c>=1 && c<=9); }
 30
 31 bool Input()
 32 {
 33     if(scanf("%s%d%d%s%d%d", name, &r0, &c0, &c, &r2, &c2)!=6)    return false;
 34     printf("%s\n", name);
 35     dir = dir_id(c[0]);
 36     r1 = r0 + dr[dir];
 37     c1 = c0 + dc[dir];
 38
 39     memset(has_edge, false, sizeof(has_edge));
 40     int a, b;
 41     while(scanf("%d", &a))
 42     {
 43         if(a == 0)    break;
 44         scanf("%d", &b);
 45         while(scanf("%s", s))
 46         {
 47             if(s[0] == ‘*‘)    break;
 48             int tempd = dir_id(s[0]);
 49             for(int i = 1; i < strlen(s); ++i)
 50                 has_edge[a][b][tempd][turn_id(s[i])] = true;
 51         }
 52     }
 53     return true;
 54 }
 55
 56 Node walk(const Node& u, int turn)
 57 {
 58     int dir = u.dir;
 59     if(turn == 1)    dir = (dir + 3) % 4;
 60     if(turn == 2)    dir = (dir + 1) % 4;
 61     return Node(u.r + dr[dir], u.c + dc[dir], dir);
 62 }
 63
 64 void print_ans(Node u)
 65 {
 66     vector<Node> nodes;
 67     for(;;)
 68     {
 69         nodes.push_back(u);
 70         if(d[u.r][u.c][u.dir] == 0)    break;
 71         u = p[u.r][u.c][u.dir];
 72     }
 73     nodes.push_back(Node(r0, c0, dir));
 74
 75     int cnt = 0;
 76     for(int i = nodes.size()-1; i >= 0; --i)
 77     {
 78         if(cnt % 10 == 0)    printf(" ");
 79         printf(" (%d,%d)", nodes[i].r, nodes[i].c);
 80         if(++cnt % 10 == 0)    printf("\n");
 81     }
 82     if(nodes.size() % 10 != 0)    printf("\n");
 83 }
 84
 85 void solve()
 86 {
 87     queue<Node> q;
 88     memset(d, -1, sizeof(d));
 89     Node u(r1, c1, dir);
 90     d[r1][c1][dir] = 0;
 91     q.push(u);
 92     while(!q.empty())
 93     {
 94         Node u = q.front();    q.pop();
 95         if(u.r == r2 && u.c == c2)
 96         {
 97             print_ans(u);
 98             return;
 99         }
100         for(int i = 0; i < 3; ++i)
101         {
102             Node v = walk(u, i);
103             if(has_edge[u.r][u.c][u.dir][i] && inside(v.r, v.c) && d[v.r][v.c][v.dir] < 0)
104             {
105                 d[v.r][v.c][v.dir] = d[u.r][u.c][u.dir] + 1;
106                 p[v.r][v.c][v.dir] = u;
107                 q.push(v);
108             }
109         }
110     }
111     puts("  No Solution Possible");
112 }
113
114 int main(void)
115 {
116     #ifdef LOCAL
117         freopen("816in.txt", "r", stdin);
118     #endif
119
120     while(Input())
121     {
122         solve();
123     }
124
125     return 0;
126 }

代码君

时间: 2024-08-10 17:11:20

UVa (一道比较复杂的广搜) 816 Abbott’s Revenge的相关文章

uva 816 - Abbott&#39;s Revenge(有一点难度的bfs迷宫题目)

就是典型的bfs,但这道题目的难点在于其条件的读取和判断并不简单,需要想办法来读取条件,也需要想办法来判断在每个点处能不能满足向下继续走的条件. #include<cstdio> #include<cstring> #include<string> #include<queue> #include<iostream> #include<algorithm> using namespace std; struct note { int

UVA 816 - Abbott&#39;s Revenge(BFS)

UVA 816 - Abbott's Revenge 题目链接 题意:一个迷宫,每个点限制了从哪一方向来的,只能往左右前走,然后问起点到终点的最短路径 思路:BFS,每个点拆成4个方向的点,对应能走的方向建图跑一下bfs即可 代码: #include <cstdio> #include <cstring> #include <vector> #include <queue> #include <algorithm> using namespace

UVA 816 Abbott’s Revenge

bfs求最短路,递归打印最短路的具体路径: 难点: 当前状态和转弯方式很复杂,要仔细处理: 递归打印:用一个数组存储路径中结点的前一个节点,递归查找 (bfs无法确定下一个结点,但对于没一个结点,它的上一个结点是确定的!) ps:输出因为太懒不想处理所以按书上打的:递归打印理解有点麻烦... 1 #include <iostream> 2 #include <cstring> 3 #include <algorithm> 4 #include <queue>

Uva 816 Abbott&#39;s Revenge(BFS)

#include<cstdio>#include<cstring>#include<vector>#include<queue>using namespace std; struct Node{ int row,col,dir; Node(int row=0,int col=0,int dir=0):row(row),col(col),dir(dir){}}; const char * dirs="NESW";const char * t

UVA 816 -- Abbott&#39;s Revenge(BFS求最短路)

 UVA 816 -- Abbott's Revenge(BFS求最短路) 有一个 9 * 9 的交叉点的迷宫. 输入起点, 离开起点时的朝向和终点, 求最短路(多解时任意一个输出即可).进入一个交叉点的方向(用NEWS表示不同方向)不同时, 允许出去的方向也不相同. 例如:1 2 WLF NR ER * 表示如果 进去时朝W(左), 可以 左转(L)或直行(F), 如果 朝N只能右转(R) 如果朝E也只能右转.* 表示这个点的描述结束啦! 输入有: 起点的坐标, 朝向, 终点的坐标.然后是各个

The Monocycle,UVa 10047(状态记录广搜)

题目链接:http://acm.hust.edu.cn/vjudge/problem/36951 题意: 一辆独轮车,他的车轮每72度变一个颜色(蓝,白,绿,黑,红)..每秒有3种操作,左转,右转,或者前进,前进的时候车轮的颜色会向前进一色.要求出到终点且车轮触底颜色为蓝色的最小时间. 初始时间为轮子触底为蓝,朝向北 源代码: #include<iostream> #include<cstdio> #include<cstring> #include<queue&

poj 3984:迷宫问题(广搜,入门题)

迷宫问题 Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7635   Accepted: 4474 Description 定义一个二维数组: int maze[5][5] = { 0, 1, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 0, 0, 0, 0, 1, 0, }; 它表示一个迷宫,其中的1表示墙壁,0表示可以走的路,只能横着走或竖着走,不能斜着走,要

UVA 122 Trees on the level 二叉树 广搜

题目链接: https://vjudge.net/problem/UVA-122 题目描述: 给你一种二叉树的构造方法, 让你逐层输出二叉树的节点值, 如果不能够则输出"not complete" 解题思路: 这道题就是硬搞就可以了, 参考紫书去做的, 首先处理输入就是非常麻烦的事情, 用到了sscanf就会轻松很多, 看来C中还是有很多很多的好用的标准库函数可以拿来用的, 例如还有本题中的strchr() , 处理完输入, 然后就去构造数. 然后广搜一遍即可 代码: #include

一道广搜寻路题

同样是在qq群里看到的题目,想了好久算法,实现也用了很久. 关于题目首先看图: 总的来说,就是一个二维迷宫的寻路,迷宫中有对应的钥匙和刺,每走一步会消耗1点Hp,当走到刺上时会额外消耗100点hp,持有对应颜色的钥匙通过刺时不用额外消耗Hp. 给予起点和终点的坐标,,输出移动方式,让人物抵达终点所消耗的Hp尽可能的小. 例子: 3 3 1..a##A...1 13 3这个是输入数据 第一个代表高 第二个宽 第三个是钥匙和陷阱的对数 .代表平地 #代表墙 小写字母是钥匙 大写字母是对应的陷阱输出为