Description

吃豆人是一款非常经典的游戏,游戏中玩家控制吃豆人在地图上吃光所有豆子,并且避免被怪物抓住。

这道题没有怪物,将游戏的画面分成n*m的格子,每格地形可能为空地或者障碍物,吃豆人可以在空地上移动,吃豆人每移动一格需要1s时间,并且只能朝上下左右四个方向移动,特别的是吃豆人还能吐出舌头,舌头每移动一格需要0.1s时间,舌头只可以走直线。不必考虑吃豆人转身所需要的时间。

举例,吃豆人在(1,1)坐标,而豆子在(1,5)坐标,并且中间没有障碍物,此时朝豆子方向吐舌头~,经过0.8s就可以吃到豆子(来回各0.4s,吐出去的舌头要缩回来的嘛)。

游戏中还有加速道具,一旦得到加速道具,吃豆人就获得2倍移动速度,吐舌头的速度没有增加,即走1格用0.5s。现在地图上有且只有一颗豆子。游戏中有.代表空地;X表示障碍,吃豆人不能越过障碍;B代表豆子;S代表加速道具,并且地图上道具总数不超过1个,道具所在的位置为空地,得到道具后立即使用,道具立即消失,地形变为空地,不能用舌头去取道具;P表示吃豆人,吐舌头的时候吃豆人不能移动。

Input

输入包含多组数据。输入第一行有两个个整数n,m(2<=n,m<=20),接着一个n*m的地图矩阵。

对于50%的数据,地图上没有道具。

Output

输出一行,最快用多少s吃到豆子,结果保留1位小数,如果吃不到,输出-1。

Sample Input

2 2
XP
B.
3 2
XP .
S B.

Sample Output

1.2
1.7
 
思路:舌头是不能越过障碍的。由于道具最多只有一个,我们分两种情况,一是不拿道具,那么可先BFS出起点到各个点的最短距离,然后再枚举每个可以走到的位置,判断在该位置吐舌头能否得到更优解。二是从起点去拿道具,然后从道具那个位置出发,也BFS出道具到每个点的最短距离,考虑到达每个位置吐舌头能否得到更优解
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
using namespace std;
const int INF = 0x3f3f3f3f;
int n, m;
char mat[][]; int px, py, bx, by, sx, sy; struct node {
int x, y;
node() {}
node(int x, int y) : x(x), y(y) {}
};
queue<node> que;
int dir[][] = { {, }, {, }, {, -}, {-, } };
int md[][][], vis[][];
void init() {
memset(md, INF, sizeof md);
sx = sy = -;
for(int i = ; i < n; ++i)
for(int j = ; j < m; ++j) {
if(mat[i][j] == 'P') px = i, py = j;
if(mat[i][j] == 'B') bx = i, by = j;
if(mat[i][j] == 'S') sx = i, sy = j;
}
}
bool check(int x, int y) {
if(x < || x >= n || y < || y >= m || vis[x][y] || mat[x][y] == 'X') return false;
return true;
}
void BFS(int x, int y, int f) {
while(!que.empty()) que.pop();
memset(vis, , sizeof vis);
vis[x][y] = ;
que.push(node(x, y));
md[f][x][y] = ;
node u, v;
while(!que.empty()) {
u = que.front();
que.pop();
for(int i = ; i < ; ++i)
{
v.x = u.x + dir[i][];
v.y = u.y + dir[i][];
if(check(v.x, v.y)) {
que.push(v);
vis[v.x][v.y] = ;
md[f][v.x][v.y] = md[f][u.x][u.y] + ;
}
}
}
}
double ans; int to(int x, int y) {
if(bx == x) {
int y1 = min(by, y), y2 = max(by, y);
for(int i = y1 + ; i <= y2; ++i) if(mat[x][i] == 'X') return -;
return y2 - y1;
}else if(by == y) {
int x1 = min(bx, x), x2 = max(bx, x);
for(int i = x1 + ; i <= x2; ++i) if(mat[i][y] == 'X') return -;
return x2 - x1;
}
return -;
}
void solve1() {
for(int i = ; i < n; ++i) {
for(int j = ; j < m; ++j) if(md[][i][j] != INF) {
int dis = to(i, j);
if(dis == -) continue;
ans = min(ans, md[][i][j] + (double)dis * 0.1 * );
}
}
}
void solve2() {
int res = md[][sx][sy];
if(res == INF) return;
for(int i = ; i < n; ++i) {
for(int j = ; j < m; ++j) if(md[][i][j] != INF) {
int dis = to(i, j);
ans = min(ans, res + md[][bx][by] / 2.0);
if(dis != -) ans = min(ans, res + md[][i][j] / 2.0 + (double)dis * 0.1 * );
}
}
}
int main() {
//freopen("in", "r", stdin);
while(~scanf("%d%d", &n, &m)) {
for(int i = ; i < n; ++i) scanf("%s", mat[i]);
init();
BFS(px, py, );
if(sx != -)
BFS(sx, sy, ); ans = INF;
if(md[][bx][by] == INF) { puts("-1"); continue; }
solve1();
if(sx != -)
solve2();
printf("%.1f\n", ans);
}
return ;
}

Fzu2124 - 吃豆人 BFS的更多相关文章

  1. FZU 2124 吃豆人 bfs

    题目链接:吃豆人 比赛的时候写的bfs,纠结要不要有vis数组设置已被访问,没有的话死循环,有的话就不一定是最优解了.[此时先到的不一定就是时间最短的.]于是换dfs,WA. 赛后写了个炒鸡聪明的df ...

  2. FZU 2124 FOJ 2124 吃豆人【BFS】

     Problem 2124 吃豆人 Accept: 134    Submit: 575 Time Limit: 1000 mSec    Memory Limit : 32768 KB  Probl ...

  3. TurnipBit开发板DIY呼吸的吃豆人教程实例

    转载请以链接形式注明文章来源(MicroPythonQQ技术交流群:157816561,公众号:MicroPython玩家汇) 0x00前言 吃豆人是耳熟能详的可爱形象,如今我们的TurnipBit也 ...

  4. [代码]解析nodejs的require,吃豆人的故事

    最近在项目中需要对nodejs的require关键字做解析,并且替换require里的路径.一开始我希望nodejs既然作为脚本语言,内核提供一个官方的parser库应该是一个稳定可靠又灵活的渠道,然 ...

  5. Unity项目 - 吃豆人Pacman

    项目展示 Github项目地址:Pacman 涉及知识 切片制作 Animations 状态机设置,any state切换,重写状态机 按键读取进行整数距离的刚体移动 用射线检测碰撞性 渲染顺序问题 ...

  6. 利用纯css写三角形,弧度箭头,吃豆人,气泡。放大镜,标签的源码

    1. 向上三角形

  7. css吃豆人动画

    一. Css吃豆人动画 1. 上半圆:两个div,内部一个圆div,外部设置宽高截取半圆 外部div动画:animation: 动画样式 1s(时长) ease(动画先低速后快速) infinite( ...

  8. 用python的turtle作图(二)动画吃豆人

    本文是用python的turtle作图的第二篇,通过这个例子可以了解动画的原理,用python自带的turtle库制作一些小动画. 1.问题描述 在上一篇"用python的turtle作图( ...

  9. Pac-Man 吃豆人

    发售年份 1980 平台 街机 开发商 南梦宫(Namco) 类型 迷宫 https://www.youtube.com/watch?v=dScq4P5gn4A

随机推荐

  1. <<< Tomcat运行提示The server does not support version 3.0

    导入了一个项目,运行Tomcat出现此错误大致由于当前导入项目服务器不支持J2ee版本 原因是导入项目的Tomcat版本是6.0,Tomcat 6.0仅支持到Servlet 2.5,而此时项目是3.0 ...

  2. bootstrap 水平表单

    <form class="form-horizontal" role="form"> <div class="form-group& ...

  3. Zabbix 监控 Nginx(四)

    简介: 如何使用 Zabbix 监控 Nginx 状态 ? 1.获取 Nginx 状态( HTTP Stub Status ) [root@localhost ~]# /apps/product/ng ...

  4. App- 借书趣

    借书趣是一款方便用户在上海图书馆借书助手应用.通过扫描条码,导入豆瓣想读等手段可以方便管理想读想借的书目应用通过上图的接口和一些算法帮助用户生成借书单,规划用户可以去上图的哪个分馆可以借到最多想要阅读 ...

  5. 利用session_set_save_handler()函数将session保存到MySQL数据库中

    PHP保存session默认的是采用的文件的方式来保存的,这仅仅在文件的空间开销很小的windows上是可以采用的,但是如果我们采用uinx或者是liux上的文件系统的时候,这样的文件系统的文件空间开 ...

  6. PHP读写大“二进制”文件,不必申请很大内存(fopen、fread、fwrite、fclose)

    <?php /** * 读写大二进制文件,不必申请很大内存 * 只有读取到内容才创建文件 * 保证目录可写 * * @param string $srcPath 源文件路径 * @param s ...

  7. iOS开发——高级篇——iPad开发、iPad开发中的modal

    一.iPad简介 1.什么是iPad一款苹果公司于2010年发布的平板电脑定位介于苹果的智能手机iPhone和笔记本电脑产品之间跟iPhone一样,搭载的是iOS操作系统 2.iPhone和iPadi ...

  8. Maven 入门 (1)—— 安装

    Maven 入门 (1)—— 安装 http://blog.csdn.net/kakashi8841/article/details/17371837 1.下载maven安装包 http://mave ...

  9. 修改hosts文件,修改后不生效怎么办

    当你在打开浏览器的情况下修改hosts文件时,关闭浏览器时系统才会释放掉hosts文件占用的那部分内存,我们再次打开浏览器访问就发现已经生效了. ps:ipconfig /flushdns     # ...

  10. myecplise 中文乱码

    一.设置新建常见文件的默认编码格式,也就是文件保存的格式. 在不对MyEclipse进行设置的时候,默认保存文件的编码,一般跟简体中文操作系统(如windows2000,windowsXP)的编码一致 ...