BZOJ4651 NOI2016网格(割点)
首先显然可以通过孤立角落里的跳蚤使其不连通,所以只要有解答案就不会大于2。同样显然的一点是当且仅当跳蚤数量<=2且连通时无解。做法其实也很显然了:特判无解,若跳蚤不连通输出0,否则看图中是否无割点(即点双连通),若无答案为2,否则为1。
现在的问题是这个图实在是太大了。正常的离散化可能仍然需要留下c2个点。这个时候发现部分分很足于是我们就可以弃疗了。
比较直观的一点是附近没有蛐蛐的跳蚤不太可能被割开。显然只有周围八连通有蛐蛐的位置才可能成为割点。那么要判断其是否是割点还需要再取周围一圈。所以取出每个蛐蛐的周围两圈跳蚤放在一张图里,之间四连通的连边。此时若有某个蛐蛐周围两圈的跳蚤处于不同连通块,则说明跳蚤本身就不连通;否则tarjan求一发割点就可以了。注意这里割点必须与蛐蛐八连通(或在边界)才是原图的割点,正确性是显然的,但好像没想明白不这么干会有什么问题。
map被卡常习惯了。bzoj过了,luoguT两个点。
upd:莫名其妙的把一个完全能用数组的东西用map存了。然后对于点的初始化在新建点的时候进行,不要直接memset。就能过掉了。
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<map>
using namespace std;
#define ll long long
#define N 100010
char getc(){char c=getchar();while ((c<'A'||c>'Z')&&(c<'a'||c>'z')) c=getchar();return c;}
int gcd(int n,int m){return m==?n:gcd(m,n%m);}
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
int T,n,m,c,dfn[N<<],low[N<<],cnt,tot,t,p[N<<],id[][],fa[N<<];
bool tag[N<<];
struct data
{
int x,y;
bool operator <(const data&a) const
{
return x<a.x||x==a.x&&y<a.y;
}
}a[N];
map<data,int> f;
int wx[]={,,,-},wy[]={,,-,};
struct data2{int to,nxt;
}edge[N<<];
int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
void addedge(int x,int y){t++;edge[t].to=y,edge[t].nxt=p[x],p[x]=t;}
bool tarjan(int k,int from)
{
dfn[k]=low[k]=++tot;int son=;
for (int i=p[k];i;i=edge[i].nxt)
if (edge[i].to!=from)
{
if (dfn[edge[i].to]) low[k]=min(low[k],dfn[edge[i].to]);
else
{
if (tarjan(edge[i].to,k)) return ;
son++;low[k]=min(low[k],low[edge[i].to]);
if (from!=-&&low[edge[i].to]>=dfn[k]||son>&&from==-) if (tag[k]) return ;
}
}
return ;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("bzoj4651.in","r",stdin);
freopen("bzoj4651.out","w",stdout);
const char LL[]="%I64d\n";
#else
const char LL[]="%lld\n";
#endif
T=read();
while (T--)
{
n=read(),m=read(),c=read();f.clear();
for (int i=;i<=c;i++) a[i].x=read(),a[i].y=read(),f[a[i]]=-;
if (1ll*n*m-c<=) {cout<<-<<endl;continue;}
if (1ll*n*m-c==)
{
int vx[],vy[],t=-;
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
if (f.find((data){i,j})==f.end()) t++,vx[t]=i,vy[t]=j;
if (vx[]==vx[]&&abs(vy[]-vy[])==||vy[]==vy[]&&abs(vx[]-vx[])==) cout<<-<<endl;
else cout<<<<endl;
continue;
}
cnt=;t=;
for (int i=;i<=c;i++)
{
for (int x=-;x<=;x++)
for (int y=-;y<=;y++)
if (a[i].x+x>=&&a[i].x+x<=n&&a[i].y+y>=&&a[i].y+y<=m)
{
if (f.find((data){a[i].x+x,a[i].y+y})==f.end())
f[(data){a[i].x+x,a[i].y+y}]=++cnt,id[x+][y+]=cnt,fa[cnt]=cnt,tag[cnt]=p[cnt]=dfn[cnt]=;
else id[x+][y+]=f[(data){a[i].x+x,a[i].y+y}];
if (a[i].x+x==||a[i].x+x==n||a[i].y+y==||a[i].y+y==m||abs(x)<=&&abs(y)<=) tag[id[x+][y+]]=;
}
else f[(data){a[i].x+x,a[i].y+y}]=-,id[x+][y+]=-;
for (int x=;x<=;x++)
for (int y=;y<=;y++)
for (int k=;k<;k++)
if (x+wx[k]>=&&x+wx[k]<=&&y+wy[k]>=&&y+wy[k]<=&&~id[x][y]&&~id[x+wx[k]][y+wy[k]])
addedge(id[x][y],id[x+wx[k]][y+wy[k]]),fa[find(id[x+wx[k]][y+wy[k]])]=find(id[x][y]);
}
bool flag=;
for (int i=;i<=c;i++)
{
int t=-;
for (int x=-;x<=;x++)
for (int y=-;y<=;y++)
{
int p=f[(data){a[i].x+x,a[i].y+y}];
if (~p) if (t==-) t=find(p);else if (t!=find(p)) {flag=;break;}
}
if (!flag) break;
}
if (!flag) {cout<<<<endl;continue;}
if (n==||m==) {cout<<<<endl;continue;}
tot=;
for (int i=;i<=cnt;i++)
if (!dfn[i]&&tarjan(i,-)) {cout<<<<endl;flag=;break;}
if (flag) cout<<<<endl;
}
return ;
}
BZOJ4651 NOI2016网格(割点)的更多相关文章
- [UOJ#220][BZOJ4651][Noi2016]网格
[UOJ#220][BZOJ4651][Noi2016]网格 试题描述 跳蚤国王和蛐蛐国王在玩一个游戏. 他们在一个 n 行 m 列的网格上排兵布阵.其中的 c 个格子中 (0≤c≤nm),每个格子有 ...
- [BZOJ4651][NOI2016]网格(Tarjan)
下面直接给出结论,相关证明见官方题解. 1.若跳蚤数不超过1或仅有两只跳蚤且相邻,则答案为-1. 2.若跳蚤形成的连通块个数大于1,则答案为0. 3.若跳蚤之间建图存在割点,则答案为1. 4.否则为2 ...
- BZOJ4651 & 洛谷1173 & UOJ220:[NOI2016]网格——题解(附debug数据)
https://www.lydsy.com/JudgeOnline/problem.php?id=4651 https://www.luogu.org/problemnew/show/P1173#su ...
- BZOJ4651/UOJ220 [Noi2016]网格
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
- 并不对劲的bzoj4651:loj2084:uoj220:p1173:[NOI2016]网格
题目大意 有一个\(n*m\)(\(n,m\leq10^9\))的网格,每个格子是空地或障碍(\(障碍数\leq10^5\)) 定义两块空地连通,当且仅当它们是"相邻的两块空地"或 ...
- [Noi2016]网格
来自FallDream的博客,未经允许,请勿转载,谢谢. 跳蚤国王和蛐蛐国王在玩一个游戏. 他们在一个 n 行 m 列的网格上排兵布阵.其中的 c 个格子中 (0≤c≤nm),每个格子有一只蛐蛐, ...
- 洛谷P1173 [NOI2016]网格
这个码量绝对是业界大毒瘤...... 300行,6.5k,烦的要死...... 题意:给你一个网格图,里面有0或1.你需要把一些0换成1使得存在某两个0不四联通.输出最小的换的数量.无解-1. n,m ...
- UOJ220 [NOI2016] 网格 【割顶】【并查集】
题目分析: 答案显然只有{-1,0,1,2}四种. 对于答案等于-1的情况,只有两种情况,一种是只剩一只跳蚤,另一种是只剩两只跳蚤且他们四连通,这个很好判. 对于答案等于0的情况,那说明联通块大于1, ...
- NOI2016
luoguP1712 [NOI2016]区间 这是一道送分题. 对于我这种每天抄题解不动脑子思维僵化得厉害的智障选手就是送命题. 一直在想端点排序各种Treap搞... 正解: 已知一些区间,如何判断 ...
随机推荐
- VMware下CentOS7安装后,还原虚拟网络后,敲ifconfig不显示局域网ip解决方法
VMware下CentOS7安装后,还原虚拟网络后,敲ifconfig不显示局域网ip,没有出现eth0网卡,不能上网,SSH不能连接,输入ifconfig后如下图: 解决方法: 1.编辑网卡的配置文 ...
- centos7中vsftp的搭建
开启vsftpd:service vsftpd start关闭vsftp:service vsftpd stop 安装vsftpd: yum -y install vsftpd 建立vsftpd帐号: ...
- Hive(1)-基本概念
一. 什么是Hive Hive:由Facebook开源用于解决海量结构化日志的数据统计. Hive是基于Hadoop的一个数据仓库工具,可以将结构化的数据文件映射为一张表,并提供类SQL查询功能. 本 ...
- echarts实用小技巧,控制字符串长度,限定整数等
限定横坐标文本字符长度 xAxis : [ axisLabel:{ formatter: function (value) { var maxlength=6; if (value.length> ...
- wamp调用ICE中间件
wamp调用ICE中间件 wamp 是集成开发包,我的wamp中的php 为5.3.10 ,经过3天艰苦奋战,终于在phpinfo()中看到了 ICE 出现了.. OK,最新的ice 为 3.5.1 ...
- java web项目使用ant编译将不同的功能代码打包成jar,进而分局点将项目打包成不同的tar.gz包进而部署
使用ant可以轻松的将一个项目分离代码,直接打包成不同需求的tar.gz包使用 1.build.properties (属性) version.num=1.0 #版本信息 2.build.xml (a ...
- python2.7练习小例子(六)
6):题目:斐波那契数列. 程序分析:斐波那契数列(Fibonacci sequence),又称黄金分割数列,指的是这样一个数列:0.1.1.2.3.5.8.13.21.34.……. ...
- [答网友问]让GridLength支持动画
原文:[答网友问]让GridLength支持动画 [答网友问]WPF中让GridLength类型支持动画 ...
- SPOJ SUBLEX
SUBLEX - Lexicographical Substring Search 链接 题意 求第k小的子串.相同的算一个. 分析 建立后缀自动机,在后缀自动机上从一个点经过trans,到另一个点, ...
- PowerPC为什么会没落,我自己的反思学习总结
PowerPC为什么会没落? Intel通过酷睿,拉大优势,使得苹果抛弃PowerPC PowerPC当年异常强大,曾经是Xbox的处理器,是Mac的处理器,占据了服务器的半壁江山:当时刚刚接触C ...