题意:给一个二维地图,每个点为障碍或者空地,有一个机器人有三种操作:1、向前走;2、左转90度;3、右转90度。现给定起点和终点,问到达终点最短路的条数。

思路:一般的题目只是求最短路的长度,但本题还要求出相应的条数。比赛时只记录最少的步数,却没有记录以最少步数到达该点的的条数,让他们一直入队.......铁定tle.......

只要记录好到达该点最少步数的条数,减少了很多重复入队..........

#include <cstdio>
#include <cstring>
#include <iostream>
#include <string>
#include <algorithm>
#include <queue>
#define MAX 1111
#define INF 0x7FFFFFFF
using namespace std; int n,m,mod;
struct node {
int x,y,dir;
} st,end; queue <node> q;
char map[MAX][MAX];
int num[MAX][MAX][4]; //步数
int cnt[MAX][MAX][4]; //多少种路径以该方向到达该点
int dx[] = {-1,0,1,0};
int dy[] = {0,1,0,-1}; void init() {
while(!q.empty()) q.pop();
memset(num,-1,sizeof(num));
memset(cnt,0,sizeof(cnt));
} int getdir(char c) {
if(c == 'N') return 0;
if(c == 'E') return 1;
if(c == 'S') return 2;
if(c == 'W') return 3;
} bool ok(int x,int y) {
if(x >= 0 && x < n && y >= 0 && y < m && map[x][y] == '.') return true;
return false;
} void bfs() {
num[st.x][st.y][st.dir] = 0;
cnt[st.x][st.y][st.dir] = 1;
q.push(st);
while(! q.empty()) {
node t = q.front();
q.pop();
node tt;
//cout << t.x << ' ' << t.y << ' ' << num[t.x][t.y][t.dir] << endl;
for(int i=0; i<4; i++) { //转方向
if(abs(t.dir - i) == 1 || abs(t.dir - i) == 3) {
tt = t;
if(num[t.x][t.y][i] == -1) {
num[t.x][t.y][i] = num[t.x][t.y][t.dir] + 1;
cnt[t.x][t.y][i] = cnt[t.x][t.y][t.dir];
tt.dir = i;
q.push(tt);
} else if(num[t.x][t.y][i] == num[t.x][t.y][t.dir] + 1) {
cnt[t.x][t.y][i] = (cnt[t.x][t.y][i] + cnt[t.x][t.y][t.dir]) % mod;
}
}
}
tt = t;
tt.x += dx[tt.dir];
tt.y += dy[tt.dir];
if(ok(tt.x,tt.y)) {
if(num[tt.x][tt.y][tt.dir] == -1) { //移动
num[tt.x][tt.y][tt.dir] = num[t.x][t.y][t.dir] + 1;
cnt[tt.x][tt.y][tt.dir] = cnt[t.x][t.y][t.dir];
q.push(tt);
} else if(num[tt.x][tt.y][tt.dir] == num[t.x][t.y][t.dir] + 1) {
cnt[tt.x][tt.y][tt.dir] = (cnt[tt.x][tt.y][tt.dir] + cnt[t.x][t.y][t.dir]) % mod;
}
}
}
} void solve() {
int ans = INF;
for(int i=0; i<4; i++) {
if(num[end.x][end.y][i] != -1) {
ans = min(ans,num[end.x][end.y][i]);
}
}
if(ans == INF) {
printf("%d -1\n",mod);
return ;
}
int sum = 0;
for(int i=0; i<4; i++) {
if(num[end.x][end.y][i] == ans) sum = (sum + cnt[end.x][end.y][i]) % mod;
}
printf("%d %d\n",mod,sum);
}
int main() {
char c;
int ca = 1;
while(scanf("%d%d%d",&n,&m,&mod) != EOF) {
if(mod == 0) break;
init();
for(int i=0; i<n; i++) scanf("%s",map[i]);
scanf("%d%d%d%d",&st.x,&st.y,&end.x,&end.y);
scanf(" %c",&c);
st.dir = getdir(c);
bfs();
printf("Case %d: ",ca++);
solve();
}
return 0;
}

HDU 4166 & BNU 32715 Robot Navigation (记忆化bfs)的更多相关文章

  1. HDU 4960 Another OCD Patient(记忆化搜索)

    HDU 4960 Another OCD Patient pid=4960" target="_blank" style="">题目链接 记忆化 ...

  2. 随手练——HDU 1078 FatMouse and Cheese(记忆化搜索)

    http://acm.hdu.edu.cn/showproblem.php?pid=1078 题意: 一张n*n的格子表格,每个格子里有个数,每次能够水平或竖直走k个格子,允许上下左右走,每次走的格子 ...

  3. HDU 4628 Pieces(状态压缩+记忆化搜索)

    http://acm.hdu.edu.cn/showproblem.php?pid=4628 题意:给个字符窜,每步都可以删除一个字符窜,问最少用多少步可以删除一个字符窜分析:状态压缩+记忆化搜索  ...

  4. HDU 4597 Play Game (DP,记忆化搜索,博弈)

    题意:Alice和Bob玩一个游戏,有两个长度为N的正整数数字序列,每次他们两个,只能从其中一个序列,选择两端中的一个拿走.他们都希望可以拿到尽量大的数字之和, 并且他们都足够聪明,每次都选择最优策略 ...

  5. HDU ACM 1078 FatMouse and Cheese 记忆化+DFS

    题意:FatMouse在一个N*N方格上找吃的,每一个点(x,y)有一些吃的,FatMouse从(0,0)的出发去找吃的.每次最多走k步,他走过的位置能够吃掉吃的.保证吃的数量在0-100.规定他仅仅 ...

  6. HDU 1331 Function Run Fun(记忆化搜索)

    Problem Description We all love recursion! Don't we? Consider a three-parameter recursive function w ...

  7. HDU - 5001 Walk(概率dp+记忆化搜索)

    Walk I used to think I could be anything, but now I know that I couldn't do anything. So I started t ...

  8. BNU 25593 Prime Time 记忆化dp

    题目链接:点击打开链接 题意: 一个游戏由3个人轮流玩 每局游戏由当中一名玩家选择一个数字作为開始 目的:获得最小的得分 对于当前玩家 O .面对 u 这个数字 则他的操作有: 1. 计分 u +1 ...

  9. HDU 3652 B-number(数位dp&amp;记忆化搜索)

    题目链接:[kuangbin带你飞]专题十五 数位DP G - B-number 题意 求1-n的范围里含有13且能被13整除的数字的个数. 思路 首先,了解这样一个式子:a%m == ((b%m)* ...

随机推荐

  1. poj 2114 Boatherds 树的分治

    还是利用点的分治的办法来做,统计的办法不一样了,我的做法是排序并且标记每个点属于哪颗子树. #include <iostream> #include <cstdio> #inc ...

  2. 构件图(Component Diagram)—UML图(八)

    构件图是显示代码自身结构的实现级别的图表.构件图由诸如源码文件.二进制代码文件.可运行文件或动态链接库 (DLL) 这种构件构成,并通过依赖关系相连接 以下这张图介绍了构件图的基本内容: 以下这张图是 ...

  3. ArrayList集合-[习题]--C#

    :向集合中添加10个元素,计算平均值,求最大.最小值. ; list.AddRange(, , , , , , , , }); int Max, Min; Max = Min = (]; ; i &l ...

  4. oracle 11gR2默认密码修改

    很久以前装了Oracle,今天终于下决心要学一学了,结果一上午的时间就贡献给如何连接数据库上了 忘记了安装时设置的用户名和密码怎么办?查了下网上的资料,终于解决了! 方法一: 首先进入sqlplus: ...

  5. Eclipse使用技巧总结(三)

    十六.快速关闭窗口   关闭当前打开窗口 Ctrl + W       关闭当前打开的所有窗口 Ctrl +Shift +F4 十九.重命名  F2 二十.快速回到上次编辑点  Ctrl + Q 二十 ...

  6. 通过xml生成word文档

    Xml生成word总结 使用xml生成word的基本步骤在<使用xslt转化xml数据形成word文档导出.doc>中说明比较清楚了.但是其中的细节并未说到,因此自己折腾了两天总算成功了. ...

  7. JQuery遍历json数组的3种方法

    这篇文章主要介绍了JQuery遍历json数组的3种方法,本文分别给出了使用each.for遍历json的方法,其中for又分成两种形式,需要的朋友可以参考下 一.使用each遍历 $(functio ...

  8. PHP高级编程SPL

    这几天,我在学习PHP语言中的SPL. 这个东西应该属于PHP中的高级内容,看上去非常复杂,可是非常实用,所以我做了长篇笔记.不然记不住,以后要用的时候,还是要从头学起. 因为这是供自己參考的笔记,不 ...

  9. Android漫游记(4)---.so文件动态调试一例

    Android平台的动态调试一直以来是个困扰我等Coder的头疼问题,特别是对于本地的动态调试支持.能够说是"弱智"级别的,不知道Google的新版NDK和新出的Android S ...

  10. HDU 1425 sort 题解

    选择出数列中前k个最大的数. 这里由于数据特殊.所以能够使用hash表的方法: #include <cstdio> #include <algorithm> #include ...