题面

考虑双向广搜。

我们需要记录男孩和女孩的当前位置,并且每次都进行扩展。

记录一个数组 \(st[i][j]\) 。

  • 如果 \(st[i][j]=0\) ,说明 \((i,j)\) 还没有被男孩和女孩经过;
  • 如果 \(st[i][j]=1\) ,说明 \((i,j)\) 被男孩经过了;
  • 如果 \(st[i][j]=2\) ,说明 \((i,j)\) 被女孩经过了。

如果当前扩展的是男孩,且当前的位置是 \((x,y)\) ,那么当 \(st[x][y]=2\) 时直接返回当前扩展的层数即可;

如果当前扩展的是女孩,那么当 \(st[x][y]=1\) 时直接返回就可以了。

题目中有一句话:

在第 \(k\) 秒后所有与鬼的曼哈顿距离不超过 \(2k\) 的位置都会被鬼占领。

我们根据这句话的提示判断该点是否要扩展即可。

其实双队列来实现双向广搜还是比较易懂的。

#include <bits/stdc++.h>
#define DEBUG fprintf(stderr, "Passing [%s] line %d\n", __FUNCTION__, __LINE__)
#define itn int
#define gI gi using namespace std; inline int gi()
{
int f = 1, x = 0; char c = getchar();
while (c < '0' || c > '9') {if (c == '-') f = -1; c = getchar();}
while (c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return f * x;
} const int maxn = 803; int t, n, m;
char s[maxn][maxn];
int st[maxn][maxn];
pair <int, int> boy, girl, ghost[5]; const int dx[] = {0, 1, 0, -1}, dy[] = {1, 0, -1, 0}; inline bool check(int x, int y, int step)
{
if (x <= 0 || x > n || y <= 0 || y > n || s[x][y] == 'X') return false; //出界了或者走到了墙
for (int i = 1; i <= 2; i+=1)
if (abs(x - ghost[i].first) + abs(y - ghost[i].second) <= step * 2) return false; //判断与鬼的曼哈顿距离
return true;
} inline int bfs()
{
memset(st, 0, sizeof(st)); //记得数组初始化
queue <pair <int, int> > qb, qg;
int tot = 0;
for (int i = 1; i <= n; i+=1)
{
for (int j = 1; j <= m; j+=1)
{
if (s[i][j] == 'M') boy = (make_pair)(i, j);
else if (s[i][j] == 'G') girl = (make_pair)(i, j);
else if (s[i][j] == 'Z') ghost[++tot] = (make_pair)(i, j);
//记录男孩、女孩和鬼的位置
}
}
qb.push(boy); qg.push(girl);
int step = 0; //扩展的层数
while (qb.size() || qg.size()) //如果还可以扩展
{
++step;
for (int i = 1; i <= 3; i+=1) //男孩扩展 3 步
{
for (int j = 1, len = qb.size(); j <= len; j+=1) //要对队列中的每一个元素进行扩展
{
pair <int, int> p = qb.front(); qb.pop();
int x = p.first, y = p.second;
if (!check(x, y, step)) continue;
for (int k = 0; k < 4; k+=1)
{
int xx = x + dx[k], yy = y + dy[k];
if (!check(xx, yy, step)) continue;
if (st[xx][yy] == 2) return step;
if (!st[xx][yy])
st[xx][yy] = 1, qb.push((make_pair)(xx, yy));
}
}
}
for (int i = 1; i <= 1; i+=1) //女孩只要扩展一步
{
for (int j = 1, len = qg.size(); j <= len; j+=1)
{
pair <int, int> p = qg.front(); qg.pop();
int x = p.first, y = p.second;
if (!check(x, y, step)) continue;
for (int k = 0; k < 4; k+=1)
{
int xx = x + dx[k], yy = y + dy[k];
if (!check(xx, yy, step)) continue;
if (st[xx][yy] == 1) return step;
if (!st[xx][yy])
st[xx][yy] = 2, qg.push((make_pair)(xx, yy));
}
}
}
}
return -1; //无解
} int main()
{
//freopen(".in", "r", stdin);
//freopen(".out", "w", stdout);
t = gi();
while (t--)
{
n = gi(), m = gi();
for (int i = 1; i <= n; i+=1) scanf("%s", s[i] + 1);
printf("%d\n", bfs());
}
return 0;
}

题解【AcWing177】噩梦的更多相关文章

  1. HDU 3085 Nightmare Ⅱ(噩梦 Ⅱ)

    HDU 3085 Nightmare Ⅱ(噩梦 Ⅱ) Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Ja ...

  2. 【CSP-S膜你考】不怕噩梦 (模拟)

    不怕噩梦 题面 蚊子最近经常做噩梦,然后就会被吓醒.这可不好.. 疯子一直在发愁,然后突然有一天,他发现蚊子其实就是害怕某些事. 如果那些事出现在她的梦里,就会害怕. 我们可以假定那个害怕的事其实是一 ...

  3. [CQOI2012]模拟工厂 题解(搜索+贪心)

    [CQOI2012]模拟工厂 题解(搜索+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1327574 链接题目地址:洛谷P3161 BZOJ P26 ...

  4. 洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心)

    洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/132 ...

  5. web前端的春天 or 噩梦

    「 微信应用号可以做什么」 简单说,微信"小程序"可以为开发者提供基于微信的表单.导航.地图.媒体和位置等开发组件,让他们在微信的网页里构建一个 HTML 5 应用.同时微信还开放 ...

  6. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  7. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  8. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  9. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

随机推荐

  1. VAR 学习笔记3

    脉冲响应图及方差分析 当使用VAR模型的时候需要完成: 选择合适的变量 就是研究变量这个没有疑问 判断滞后阶数 根据AIC和SC准则,选择 为何做格兰杰因果检验 如果给定 \(x_t\) 的滞后阶数, ...

  2. Html介绍,认识html标签

    什么是网页?网页就是我们我们提前写好的代码样式经过浏览器的渲染展示出来的样式效果.其实我们常说的上网就是浏览各式各样的网页,这些网页都是由html标签组成,下面就是一个简单的网页,效果图如下: 简单看 ...

  3. Mac 终端 Tomcat 环境配置过程

    Tomcat是Apache 软件基金会(Apache Software Foundation)的Jakarta 项目中的一个核心项目,由Apache.Sun 和其他一些公司及个人共同开发而成.Tomc ...

  4. 在VS的依赖项中引用项目

    操作步骤:鼠标右击项目(注意是项目)->添加->引用->项目(在项目列表中选择需要引用的项目)->确定

  5. 【database】oracle触发器基础

    一.oracle触发器基本语法 CREATE [OR REPLACE] TRIGGER trigger_name {BEFORE | AFTER } {INSERT | DELETE | UPDATE ...

  6. Linux学习笔记:磁盘分区

    本文更新于2019-12-30.操作系统为Debian 8.9 (jessie). 以下假设新磁盘为/dev/sdb,要创建一个分区/dev/sdb1,文件系统类型为xfs.请根据实际情况,自行选择. ...

  7. mysql数据库环境配置及部分问题

    亲身经历了MySQL初学者的痛苦,把主要问题和解决方法整理一下. 一.解压版环境配置 1.把压缩包解压到某盘符下. 解压后在类似“E:\mysql-5.7.22-winx64”这个文件夹中可以看到以上 ...

  8. P1462 通往奥格瑞玛的道路【二分+Dij】

    P1462 通往奥格瑞玛的道路 提交 29.89k 通过 6.88k 时间限制 1.00s 内存限制 125.00MB 题目提供者gconeice 难度提高+/省选- 历史分数100 提交记录 查看题 ...

  9. c#实现把异常写入日志示例(异常日志)

    将异常写到日志文件中,可以在调试程序的时候知道程序发生过哪些异常,并且可以知道异常发生的位置.这点对需要进行长时间运行并调试的程序尤为有效. /// <summary> /// 将异常打印 ...

  10. ArcGIS Server Manager 重置密码

    重置忘记的密码 ArcGIS 10.5 (Windows) | 其他版本 根据以下说明为标识存储中的用户或主站点管理员帐户重置密码. 为标识存储中的用户重置密码 可按照以下步骤为标识存储中的用户重置密 ...