首页 技术 正文
技术 2022年11月14日
0 收藏 423 点赞 3,486 浏览 1903 个字

https://codeforces.com/contest/1236/problem/D

题意:一个n*m格子矩阵,放一个人偶在左上角向右走,只能在每个格子最多右转一次,有k个障碍物。求是否能够一次走完矩阵的所有非障碍物格然后停留在任意位置。

题解:在每个格子最多右转一次,相当于每个格子只能走一次,否则就出不来了。容易想到障碍物必须也是占据一些蛇形的片段,并且留下的位置刚好可以让人偶走一个蛇形的绕到中心,但是怎么判断障碍物的形状呢?百思不得其解遂看题解。题解表示观察到人偶撞墙或者撞障碍物必转向(自己走过的路也是墙)。然后暴力模拟一遍,每次在数据结构里面搜索前进方向上最近的墙/障碍物,走到它面前。至多进行n+m次。

所以选择一个结构就是set,维护行和列分别两堆set,一个障碍物同时插入行列两个set里面。然后墙/自己走过形成的墙就设置一个最值把多走的部分截断就可以了。判断答案的方法就是暴力统计走过的格子数是否等于空格数。

注:实现的时候卡在了样例4,因为会让人偶走回头路,但是也不能够简单让人偶移动0格就退出,比如一条竖线的矩阵。修复的方法是在人偶的后方一个格子塞一个障碍物。然后卡在样例9,看了dalao的代码才知道错在哪里,首先while循环的停止条件错了,不应该是d>u||r>l,应该是d>=u&&r>=l,因为是闭区间。这个改进再加上后面堵一个障碍物就可以防止原路返回了,人偶会在中间不断转圈圈。不过最简单的办法是记录人偶原地不动的次数,>=2次就break掉。

其实并不需要堵格子,记录原地不动的次数之后就不会原路返回了。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;set<int> sr[100005], sc[100005];int main() {
#ifdef KisekiPurin
freopen("KisekiPurin.in", "r", stdin);
#endif // KisekiPurin
int n, m, k;
scanf("%d%d%d", &n, &m, &k);
for(int i = 1; i <= n; ++i) {
sr[i].insert(0);
sr[i].insert(m + 1);
}
for(int i = 1; i <= m; ++i) {
sc[i].insert(0);
sc[i].insert(n + 1);
}
for(int i = 1; i <= k; ++i) {
int r, c;
scanf("%d%d", &r, &c);
sr[r].insert(c);
sc[c].insert(r);
} int u = 1, d = n, l = 1, r = m;
int x = 1, y = 1, dir = 1, stay = 0;
ll cnt = 1;
while(d >= u && r >= l) {
if(dir == 1) {///行向右
int ny = min(r, *sr[x].upper_bound(y) - 1);///r是蛇不断走的时候的边界,例如我们第一开始在第一行撞到了阻碍物,那么我们只能右转对吧,那此刻的r边界在下面就更新了,下一次在行里面就不能靠阻碍物位置了而是自己的以该r为边界,因为一个格子不能重复两次
cnt += ny - y;
if(ny == y) {
if(stay == 0)
stay = 1;
else
break;
} else
stay = 0;///如果出现原地踏步两次直接break;
y = ny; //^
++u;//上边界+1,因为该行--------->已经被通过一次了,我们下次|上来的时候是不能碰到这一行的
// |
r = y;
dir = 2;
}
else if(dir == 2) {///列向下
int nx = min(d, *sc[y].upper_bound(x) - 1);
cnt += nx - x;
if(nx == x) {
if(stay == 0)
stay = 1;
else
break;
} else
stay = 0;
x = nx;
--r;
d = x;
dir = 3;
} else if(dir == 3) {///行向左
int ny = max(l, *(--(sr[x].lower_bound(y))) + 1);
cnt += y - ny;
if(ny == y) {
if(stay == 0)
stay = 1;
else
break;
} else
stay = 0;
y = ny;
--d;
l = y;
dir = 4;
} else {///列向上
int nx = max(u, *(--(sc[y].lower_bound(x))) + 1);
cnt += x - nx;
if(nx == x) {
if(stay == 0)
stay = 1;
else
break;
} else
stay = 0;
x = nx;
++l;
u = x;
dir = 1;
}
}
puts((cnt == 1ll * n * m - k) ? "YES" : "NO");
}

  

相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:8,997
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,511
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,356
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,139
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:7,770
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:4,848