题解【AcWing177】噩梦
考虑双向广搜。
我们需要记录男孩和女孩的当前位置,并且每次都进行扩展。
记录一个数组 \(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】噩梦的更多相关文章
- HDU 3085 Nightmare Ⅱ(噩梦 Ⅱ)
HDU 3085 Nightmare Ⅱ(噩梦 Ⅱ) Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Ja ...
- 【CSP-S膜你考】不怕噩梦 (模拟)
不怕噩梦 题面 蚊子最近经常做噩梦,然后就会被吓醒.这可不好.. 疯子一直在发愁,然后突然有一天,他发现蚊子其实就是害怕某些事. 如果那些事出现在她的梦里,就会害怕. 我们可以假定那个害怕的事其实是一 ...
- [CQOI2012]模拟工厂 题解(搜索+贪心)
[CQOI2012]模拟工厂 题解(搜索+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1327574 链接题目地址:洛谷P3161 BZOJ P26 ...
- 洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心)
洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/132 ...
- web前端的春天 or 噩梦
「 微信应用号可以做什么」 简单说,微信"小程序"可以为开发者提供基于微信的表单.导航.地图.媒体和位置等开发组件,让他们在微信的网页里构建一个 HTML 5 应用.同时微信还开放 ...
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
随机推荐
- 快速建立一个Django项目
快速建立一个Django项目 版本说明 一定要先明确好使用的Python版本和所使用包的版本,避免耽误不要的时间 Python==3.6.4 Django==1.11.9 djangoresframe ...
- 信号处理之DFT、IDFT
一.DFT之前言部分 由于matlab已提供了内部函数来计算DFT.IDFT,我们只需要会调用fft.ifft函数就行: 二.函数说明: fft(x):计算N点的DFT.N是序列x的长度,即N=len ...
- opencv —— 同时识别三种颜色
要点: 1.识别一种颜色 minH = ; //色相 maxH = ; minS = ; //饱和度 maxS = ; minV = ; // inRange(原图像, 最小值的范围, 最大值的范围, ...
- ls, chgrp, chown, chmod
ls命令 [root@client ~]# ls -la 总用量 dr-xr-x---. root root 2月 : . dr-xr-xr-x. root root 2月 : .. -rwxrwxr ...
- CentOS7安装gotoblas遇到的问题
1. 错误信息: /usr/lib64/gcc/x86_64-suse-linux/4.7/../../../../x86_64-suse-linux/bin/ld: cannot find -l-l ...
- Java连载86-List集合详解
一.List集合 1.List集合存储元素的特点: (1)有序(List集合中存储有下标):存进去是这样的顺序,取出来还是按照这个顺序取出. (2)可重复 2.深入ListJ集合 ArrayLis ...
- nginx模块之ngx_http_upstream_module
ngx_http_upstream_module 示例: http上下文: upstream upservers{ ip_hash; //根据客户端IP进行调度,每个客户端ip地址访问时每个ip生成一 ...
- Codeforces Round #592 (Div. 2) D - Paint the Tree
题目链接:https://codeforces.com/contest/1244/problem/D 题意:给你一个树,让你把树上的每个节点染成三种颜色,使得任意三个互相相邻的节点颜色都不一样(意思是 ...
- 区间dp(模板+例题)
参考博文:区间dp小结(附经典例题) 首先,什么是区间dp?它是干什么的? 先在小区间进行DP得到最优解,然后再利用小区间的最优解合并求大区间的最优解 操作往往涉及到区间合并问题 以上. 模板如下: ...
- GYCTFblacklist[堆叠注入 ]
考点:堆叠注入 handler语句代替select查询 类似于强网杯随便注 不同是过滤了alter,set等函数,不能通过改变列命或着sql预处理查询表内数据 可以使用handler语句代替selec ...