题意:给一个二维地图,每个点为障碍或者空地,有一个机器人有三种操作: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. Spark Core源代码分析: Spark任务模型

    概述 一个Spark的Job分为多个stage,最后一个stage会包含一个或多个ResultTask,前面的stages会包含一个或多个ShuffleMapTasks. ResultTask运行并将 ...

  2. 使用ffmpeg将BMP图片编码为x264视频文件,将H264视频保存为BMP图片,yuv视频文件保存为图片的代码

    ffmpeg开源库,实现将bmp格式的图片编码成x264文件,并将编码好的H264文件解码保存为BMP文件. 实现将视频文件yuv格式保存的图片格式的測试,图像格式png,jpg, gif等等測试均O ...

  3. android jsonarray

    Json数组是子元素的有序集合,每个子元素都有一个下标,可以根据下标操纵Json数组的子元素.类JsonArray是bantouyan-json库对Json数组的抽象,提供操纵Json数组的各种方法. ...

  4. Ibatis调用存储过程实现增删改以及分页查询

    1.Ibatis实现增删改操作很简单了,通常我是将某一模块的增删改功能写在一个存储过程里,通过一个标识符去区分执行增加还是修改抑或删除操作. statement: <!-- 存储过程:实现学生的 ...

  5. 一口一口吃掉Hibernate(八)——Hibernate中inverse的用法

    一.Inverse是hibernate双向关系中的基本概念.inverse的真正作用就是指定由哪一方来维护之间的关联关系.当一方中指定了“inverse=false”(默认),那么那一方就有责任负责之 ...

  6. python 求值表达式解析

    采用中缀转后缀的算法. 注意我输入的格式. #注意格式 def suffix(st): listopt=[" "] listnum=[" "] for i in ...

  7. 蓝桥杯java高职组

    标题1: 猜年龄     美国数学家维纳(N.Wiener)智力早熟,11岁就上了大学.他曾在1935~1936年应邀来中 国清华大学讲学.     一次,他参加某个重要会议,年轻的脸孔引人注目.于是 ...

  8. jQuery选择器——全新的总结方式

    jQuery 选择器的总结 用于定位的选择器: 基本选择器:(用来进行绝对定位) $("#myELement")    选择id值等于myElement的元素,id值不能重复在文档 ...

  9. android设置eclipse中的自动提示功能

    菜单window->Preferences->Java->Editor->Content Assist->Enable auto activation 选项要打上勾 (并 ...

  10. hdu 3874 Necklace(线段树)

    这道题目和我之前做过的一道3xian大牛出的题目很像,不过总的来说还是要简单一点儿. 计算区间内的值的时候如果两个值相等,只能计算其中一个. 这道题需要将所有的问题输入之后再计算,首先,对所有问题的右 ...