[NOI 2016]网格
Description
给出一张 \(n\times m\) 的网格,在其中删去 \(c\) 个格子,问至少再删去几个能使得图上存在两点不连通,或输出无解。
多组询问,询问组数 \(T\) 。
\(1\leq T\leq 20,1\leq n,m\leq 10^9,\sum c\leq 10^5\)
Solution
观察题目,容易发现答案只会在 \(-1,0,1,2\) 之间,那么就可以随机输出其中一个数,以 \(\frac{1}{4^T}\) 的概率通过这道题。
那么分类讨论。
- 如果答案为 \(-1\) ,显然只有两种情况:只有剩一个格子,或者只有两个格子且两个格子相邻。这些是可以直接特判的。
- 如果答案为 \(0\) ,显然是原图已经出现了不连通的状况。
- 若果答案为 \(1\) ,那么原图存在一个点,使得删去这个点后图不联通,即原图存在割点。
- 否则答案为 \(2\) 。
由于坐标过大,我们不能直接在原图上求解,考虑离散。其实注意到网格上留有的格子有用的只是与每个删掉的格子八连通的格子。
我们可以把这些格子留下来建图。不过有一种特殊的情况,那就是边界问题,举一个例子
.....
...00
...X1
...00
.....
如题原本是一个 \(5\times 5\) 的格子,在 \((3,5)\) 被删掉了一个格子,如果我们只考虑八连通,那么就会出现 \(\text{X}\) 是一个割点,然而在原图中并不是,所以就考虑将一个被删除的点的周围 \(5\times 5\) 的格子(共 \(24\) 个)都拿下来建新图。这样点数是 \(O(24\sum c)\) 的。
特判掉 \(-1,0\) 之后,就跑一遍 \(\text{tarjan}\) 判是否有割点即可。
Code
我用了 \(\text{map}\) ,在 \(\text{UOJ}\) 上过不去...( \(\text{bzoj}\) 过了就苟且偷生...233333...
#include <bits/stdc++.h>
#define pb push_back
using namespace std;
const int N = 100000+5;
const int w1[8] = {1, -1, 0, 0, 1, 1, -1, -1};
const int w2[8] = {0, 0, 1, -1, -1, 1, -1, 1};
void gi(int &x) {
char ch = getchar(); x = 0;
for (; ch < '0' || ch > '9'; ch = getchar());
for (; ch >= '0' && ch <= '9'; ch = getchar()) x = (x<<1)+(x<<3)+ch-48;
}
int n, m, c, u, v, idx, color[N<<3], cnt;
struct pii {
int first, second;
pii (int _first = 0, int _second = 0) {first = _first, second = _second; }
bool operator < (const pii &b) const {
return first == b.first ? second < b.second : first < b.first;
}
bool operator == (const pii &b) const {return first == b.first && second == b.second; }
};
pii a[N], t, t1;
map<pii, int>id, mp;
queue<pii>Q, P;
struct tt {int to, next; }edge[N<<6];
int path[N<<3], top, rk[N<<3], iscut[N<<3], dfn[N<<3], low[N<<3], times;
bool exist(int u, int v) {
t = a[lower_bound(a+1, a+c+1, pii(u, v))-a];
return t.first == u && t.second == v;
}
bool outof(int x, int y, int u, int d, int l, int r) {
return x < u || x > d || y < l || y > r;
}
void add(int u, int v) {edge[++top] = (tt){v, path[u]}; path[u] = top; }
bool judge() {
if (1ll*n*m-c <= 1) return false;
if (1ll*n*m-c >= 3) return true;
for (int i = 1; i <= n; i++)
for (int j = 1; j <= m; j++) {
if (!exist(i, j)) {
for (int k = 0; k < 4; k++) {
if (outof(i+w1[k], j+w2[k], 1, n, 1, m)) continue;
if (!exist(i+w1[k], j+w2[k])) return false;
}
return true;
}
}
}
void bfs(pii t) {
P.push(t); int u, v, u1, v1;
while (!P.empty()) {
t = P.front(); P.pop(); u = t.first, v = t.second; int c = color[id[t]];
for (int k = 0; k < 4; k++)
if (!outof(u1 = u+w1[k], v1 = v+w2[k], 1, n, 1, m) && !exist(u1, v1) && id.count(t1 = pii(u1, v1))) {
if (color[id[t1]]) continue;
color[id[t1]] = c; P.push(t1);
}
}
}
bool connect() {
while (!Q.empty()) {
t = Q.front(); Q.pop();
if (color[id[t]]) continue;
else color[id[t]] = ++cnt, bfs(t);
}
for (int Id = 1; Id <= c; Id++) {
int u = a[Id].first, v = a[Id].second, c = -1;
for (int i = -2; i <= 2; i++)
for (int j = -2; j <= 2; j++) {
if (outof(u+i, v+j, 1, n, 1, m) || exist(u+i, v+j)) continue;
t = pii(u+i, v+j);
if (c == -1) c = color[id[t]];
else if (c != color[id[t]]) return false;
}
}
return true;
}
void build(int u, int v) {
for (int i = -2; i <= 2; i++)
for (int j = -2; j <= 2; j++) {
if (outof(u+i, v+j, 1, n, 1, m) || exist(u+i, v+j)) continue;
t = pii(u+i, v+j);
int x = id.count(t) ? id[t] : (Q.push(t), id[t] = ++idx);
if (!outof(u+i+1, v+j, max(1, u-2), min(n, u+2), max(1, v-2), min(m, v+2)) && !exist(u+i+1, v+j)) {
t = pii(u+i+1, v+j);
int y = id.count(t) ? id[t] : (Q.push(t), id[t] = ++idx);
if (x > y) {if (!mp.count(pii(y, x))) add(x, y); add(y, x); mp[pii(y, x)] = 1; }
else {if (!mp.count(pii(x, y))) add(x, y); add(y, x); mp[pii(x, y)] = 1; }
}
if (!outof(u+i, v+j+1, max(1, u-2), min(n, u+2), max(1, v-2), min(m, v+2)) && !exist(u+i, v+j+1)) {
t = pii(u+i, v+j+1);
int y = id.count(t) ? id[t] : (Q.push(t), id[t] = ++idx);
if (x > y) {if (!mp.count(pii(y, x))) add(x, y); add(y, x); mp[pii(y, x)] = 1; }
else {if (!mp.count(pii(x, y))) add(x, y); add(y, x); mp[pii(x, y)] = 1; }
}
if (abs(i) <= 1 && abs(j) <= 1) rk[x] = 1;
}
}
void tarjan(int u, int fa) {
iscut[u] = 0; dfn[u] = low[u] = ++times; int sz = 0, v;
for (int i = path[u]; i; i = edge[i].next) {
if ((v = edge[i].to) == fa) continue;
if (!dfn[v]) {
++sz; tarjan(v, u);
if (low[v] >= dfn[u]) iscut[u] = 1;
low[u] = min(low[u], low[v]);
}else low[u] = min(low[u], dfn[v]);
}
if (fa == 0 && sz == 1) iscut[u] = 0;
}
void work() {
memset(path, top = idx = 0, sizeof(path)); id.clear(), mp.clear();
memset(dfn, times = 0, sizeof(dfn)); memset(rk, 0, sizeof(rk));
memset(color, cnt = 0, sizeof(color));
gi(n), gi(m), gi(c);
for (int i = 1; i <= c; i++) {
gi(u), gi(v); a[i] = pii(u, v);
}
sort(a+1, a+c+1); a[c+1] = pii(0, 0);
if (!judge()) {puts("-1"); return; }
for (int i = 1; i <= c; i++) build(a[i].first, a[i].second);
if (!connect()) {puts("0"); return; }
if (n == 1 || m == 1) {puts("1"); return; }
for (int i = 1; i <= idx; i++) {
if (!dfn[i]) tarjan(i, 0);
if (rk[i] && iscut[i]) {puts("1"); return; }
}
puts("2");
}
int main() {
int t; gi(t); while (t--) work();
return 0;
}
[NOI 2016]网格的更多相关文章
- [LOJ 2083][UOJ 219][BZOJ 4650][NOI 2016]优秀的拆分
[LOJ 2083][UOJ 219][BZOJ 4650][NOI 2016]优秀的拆分 题意 给定一个字符串 \(S\), 求有多少种将 \(S\) 的子串拆分为形如 AABB 的拆分方案 \(| ...
- Noi 2016
考砸只能说自己弱 Noi不是生活的全部, 人们也不会永远止步于失败. 大家加油 可以+我的qq:582744883
- 数据结构(线段树):NOI 2016 区间
[问题描述] [输入格式] [输出格式] [样例输入] 6 3 3 5 1 2 3 4 2 2 1 5 1 4 [样例输出] 2 [样例说明] [更多样例] 下载 [样例 2 输入输出] 见目录下的 ...
- NOI 2016 优秀的拆分 (后缀数组+差分)
题目大意:给你一个字符串,求所有子串的所有优秀拆分总和,优秀的拆分被定义为一个字符串可以被拆分成4个子串,形如$AABB$,其中$AA$相同,$BB$相同,$AB$也可以相同 作为一道国赛题,95分竟 ...
- NOI 2016 Day1 题解
今天写了NOI2016Day1的题,来写一发题解. T2 网格 题目传送门 Description \(T\) 次询问,每次给出一个 \(n\times m\) 的传送门,上面有 \(c\) 个位置是 ...
- 字符串(后缀自动机):NOI 2016 优秀的拆分
[问题描述] 如果一个字符串可以被拆分为 AABB 的形式,其中 A 和 B 是任意非空字符串, 则我们称该字符串的这种拆分是优秀的. 例如,对于字符串 aabaabaa,如果令 A = aab, B ...
- noi 2016 游记
先挖个坑..这回大概不会太监吧(大雾 day -2 下午起飞的飞机,晚上到了成都..把东西扔到旅馆后就组队外出觅食了... 街上人不多,逛了半天才发现一家卖本地小吃的小店. KPM:诶诶给我来碗酸辣粉 ...
- [NOI 2016]区间
Description 在数轴上有 $n$ 个闭区间 $[l_1,r_1],[l_2,r_2],...,[l_n,r_n]$.现在要从中选出 $m$ 个区间,使得这 $m$ 个区间共同包含至少一个位置 ...
- 【BZOJ 4652】【NOI 2016】循环之美
题目连接: 传送 题解: 真是一道好题…… 一: 一个分数$\frac{x}{y}$完全循环当其第一次出现时,当且仅当y与k互质,x与y互质,且y不等于1. 整数情况下y一定为1,即也满足以上判断. ...
随机推荐
- CSS 基础 例子 背景色 & 背景图片
背景简写形式 : body {background:#ffffff url('img_tree.png') no-repeat right top;} 一.背景色 background-color ...
- 4.update更新和delete删除用法
一.update更新 UserMapper.java package tk.mybatis.simple.mapper; import org.apache.ibatis.annotations.Pa ...
- UNIGUI上传文件
UNIGUI上传文件 uniGUI提供了一个文件上传控件TUniFileUpload,进行数据的导入就变得比较容易.首先将TUniFileUpload控件放置在窗体上,按下导入按钮后,执行TUniFi ...
- 利用github搭建个人maven仓库
之前看到有开源项目用了github来做maven仓库,寻思自己也做一个.研究了下,记录下. 简单来说,共有三步: deploy到本地目录 把本地目录提交到gtihub上 配置github地址为仓库地址 ...
- SoftWareHelper
SoftWareHelper 环境 Visual Studio 2017,.Net Framework 4.0 SDK GitHub源码:https://github.com/yanjinhuagoo ...
- Lerning Entity Framework 6 ------ Introduction to TPT
Sometimes, you've created a table for example named Person. Just then, you want to add some extra in ...
- mysql5.7 安装错误解决
1.5.7初始化报错 2019-04-29 21:40:34 [ERROR] Child process: /home/work/mysql/bin/mysqldterminated prematur ...
- linux 服务器脚本采集数据中文无法执行错误
问题描述:在 RHEL6 版本的服务器上使用脚本操作数据库,其中一个SQL的字段值为中文,每次使用 crontab定时 执行该脚本无法获取数据,实现预期效果,而手动执行正常. oracle clien ...
- iframe 跨域调用父级方法的方案
一.如果高层域名相同的话,可以通过document.domain来实现跨域访问 例如: 父级域名:localhost:8080 子级域名:localhost:9090 那么可以设置document.d ...
- 8 Ways to Become a Better Coder
It’s time to get serious about improving your programming skills. Let’s do it! That’s an easy career ...