题意

不想写。

题解

场上想了 1h+ 无果,一到场外就口胡出来了,我真是个 sb。

首先注意到如果 \(t\) 满足条件那么 \(t+1\) 也会满足,所以答案具有单调性,可以二分,于是现在只需要考虑对于某个特定的 \(t\) 能否满足。

注意到对于平面上的 \(u,v,w\) 三个点来说,如果 \(u\) 能到 \(v\),\(v\) 能到 \(w\) 那么 \(u\) 一定能到 \(w\)。

既然这个可达性是可以传递的,那么可以用一个并查集来将相互到的点缩成一个连通块。

如果连通块个数为 \(1\) 那么一定可以,大于 \(4\) 就一定不行,剩下的情况讨论一下。

如果连通块个数为 \(2\),那么只需要找出两个分属不同连通块的点,然后看看加一个点能不能使得这两个点互相可达即可。

如果连通块个数为 \(3\),那么只需要找出一条线段和一个点,线段的两个端点和选出来的点分属三个不同的连通块。这个时候注意到加的那个点是已经确定了的,所以直接判断一下即可。

如果连通块个数为 \(4\),那么只需要找出两条相交的线段,四个端点分属不同的连通块。这个时候新加的那个点就是线段的端点。简单判断一下可达性即可。

本题代码细节比较多,建议配合代码理解。

代码

#include<bits/stdc++.h>
using namespace std;
typedef long long int ll;
const ll MAXN=1e3+51;
struct Tuple{
ll x,y;
inline bool operator <(const Tuple &rhs)const
{
return this->x==rhs.x?this->y<rhs.y:this->x<rhs.x;
}
};
map<ll,vector<Tuple>>row,col;
ll n,totr,totc,l,r,mid,res=-1;
ll ffa[MAXN],x[MAXN],y[MAXN],bel[MAXN],rows[MAXN],cols[MAXN];
inline ll read()
{
register ll num=0,neg=1;
register char ch=getchar();
while(!isdigit(ch)&&ch!='-')
{
ch=getchar();
}
if(ch=='-')
{
neg=-1;
ch=getchar();
}
while(isdigit(ch))
{
num=(num<<3)+(num<<1)+(ch-'0');
ch=getchar();
}
return num*neg;
}
inline ll find(ll x)
{
return x==ffa[x]?x:ffa[x]=find(ffa[x]);
}
inline void merge(ll x,ll y)
{
ll fx=find(x),fy=find(y);
if(fx!=fy)
{
ffa[fy]=fx;
}
}
inline ll check(ll mid)
{
ll blk=0,p,q,r,s,lx,rx;
for(register int i=1;i<=n;i++)
{
ffa[i]=i;
}
for(register int i=1;i<=n;i++)
{
for(register int j=i+1;j<=n;j++)
{
if(x[i]==x[j]&&abs(y[i]-y[j])<=mid)
{
merge(i,j);
}
if(y[i]==y[j]&&abs(x[i]-x[j])<=mid)
{
merge(i,j);
}
}
}
for(register int i=1;i<=n;i++)
{
blk+=(find(i)==i);
}
if(blk==1)
{
return 1;
}
if(blk>4)
{
return 0;
}
for(register int i=1;i<=n;i++)
{
bel[i]=ffa[i];
}
if(blk==2)
{
for(register int i=1;i<=n;i++)
{
for(register int j=1;j<=n;j++)
{
if(bel[i]==bel[j])
{
continue;
}
if(x[i]==x[j]&&abs(y[i]-y[j])<=2*mid)
{
return 1;
}
if(y[i]==y[j]&&abs(x[i]-x[j])<=2*mid)
{
return 1;
}
if(abs(x[i]-x[j])<=mid&&abs(y[i]-y[j])<=mid)
{
return 1;
}
}
}
return 0;
}
if(blk==3)
{
vector<Tuple>v;
for(register int i=1;i<=totr;i++)
{
for(register int j=1;j<row[rows[i]].size();j++)
{
p=row[rows[i]][j-1].y,q=row[rows[i]][j].y;
bel[p]!=bel[q]?v.push_back((Tuple){p,q}):(void)1;
}
}
for(register int i=1;i<=totc;i++)
{
for(register int j=1;j<col[cols[i]].size();j++)
{
p=col[cols[i]][j-1].y,q=col[cols[i]][j].y;
bel[p]!=bel[q]?v.push_back((Tuple){p,q}):(void)1;
}
}
for(register int i=0;i<v.size();i++)
{
p=v[i].x,q=v[i].y;
for(register int j=1;j<=n;j++)
{
if(bel[p]==bel[j]||bel[q]==bel[j])
{
continue;
}
if(x[p]==x[q])
{
lx=min(y[p],y[q]),rx=max(y[p],y[q]);
if(y[j]>lx&&y[j]<rx&&abs(x[j]-x[p])<=mid)
{
if(abs(y[j]-y[p]<=mid)&&abs(y[j]-y[q])<=mid)
{
return 1;
}
}
}
else
{
lx=min(x[p],x[q]),rx=max(x[p],x[q]);
if(x[j]>lx&&x[j]<rx&&abs(y[j]-y[p])<=mid)
{
if(abs(x[j]-x[p]<=mid)&&abs(x[j]-x[q])<=mid)
{
return 1;
}
}
}
}
}
return 0;
}
if(blk==4)
{
vector<Tuple>vx,vy;
for(register int i=1;i<=totr;i++)
{
for(register int j=1;j<row[rows[i]].size();j++)
{
p=row[rows[i]][j-1].y,q=row[rows[i]][j].y;
bel[p]!=bel[q]?vx.push_back((Tuple){p,q}):(void)1;
}
}
for(register int i=1;i<=totc;i++)
{
for(register int j=1;j<col[cols[i]].size();j++)
{
p=col[cols[i]][j-1].y,q=col[cols[i]][j].y;
bel[p]!=bel[q]?vy.push_back((Tuple){p,q}):(void)1;
}
}
for(register int i=0;i<vx.size();i++)
{
for(register int j=0;j<vy.size();j++)
{
p=vx[i].x,q=vx[i].y,r=vy[j].x,s=vy[j].y,lx=x[p],rx=y[r];
if(bel[p]==bel[r]||bel[p]==bel[s])
{
continue;
}
if(bel[q]==bel[r]||bel[q]==bel[s])
{
continue;
}
if(y[r]<=min(y[p],y[q])||y[r]>=max(y[p],y[q]))
{
continue;
}
if(x[p]<=min(x[r],x[s])||x[p]>=max(x[r],x[s]))
{
continue;
}
if(abs(rx-y[p])<=mid&&abs(rx-y[q])<=mid)
{
if(abs(lx-x[r])<=mid&&abs(lx-x[s])<=mid)
{
return 1;
}
}
}
}
return 0;
}
}
int main()
{
n=read(),r=2e9;
for(register int i=1;i<=n;i++)
{
rows[++totr]=x[i]=read(),cols[++totc]=y[i]=read();
row[x[i]].push_back((Tuple){y[i],i});
col[y[i]].push_back((Tuple){x[i],i});
}
sort(rows+1,rows+totr+1),sort(cols+1,cols+totc+1);
totr=unique(rows+1,rows+totr+1)-rows-1;
totc=unique(cols+1,cols+totc+1)-cols-1;
for(register int i=1;i<=totr;i++)
{
sort(row[rows[i]].begin(),row[rows[i]].end());
}
for(register int i=1;i<=totc;i++)
{
sort(col[cols[i]].begin(),col[cols[i]].end());
}
while(l<=r)
{
mid=(l+r)>>1;
check(mid)?res=mid,r=mid-1:l=mid+1;
}
printf("%d\n",res);
}

CodeForces 1419F Rain of Fire的更多相关文章

  1. Codeforces 988F Rain and Umbrellas(DP)

    题目链接:http://codeforces.com/contest/988/problem/F 题目大意: 有三个整数a,n,m,a是终点坐标,给出n个范围(l,r)表示这块区域下雨,m把伞(p,w ...

  2. Codeforces Round #436 E. Fire(背包dp+输出路径)

    题意:失火了,有n个物品,每个物品有价值pi,必须在时间di前(小于di)被救,否则就要被烧毁.救某个物 品需要时间ti,问最多救回多少价值的物品,并输出救物品的顺序. Examples Input ...

  3. Codeforces 988F. Rain and Umbrellas

    解题思路:动态规划 遍历点i,如果从前一个点i-1走到这个点i不需要伞,则疲劳值不变dp[i] = dp[i-1]. 如果前一个点i-1走到这一个点i需要伞,则从前面找一把伞. 即遍历前面的每个点j, ...

  4. CodeForces 题目乱做

    是个补题记录. 1419 除了 F 场上都过了. CF1419A Digit Game 这题好多人 FST 啊-- 考虑如果串长为奇数那么最后操作的肯定是第一个人,串长为偶数的最后操作的肯定是第二个, ...

  5. little alchemy攻略

    一个造物游戏: acidrain=rain+smoke airlplain=metal+bird alcohol=fruit+time algae=plant+water allergy=dust+h ...

  6. [Codeforces 864E]Fire

    Description Polycarp is in really serious trouble — his house is on fire! It's time to save the most ...

  7. Codeforces Round #486 (Div. 3) F. Rain and Umbrellas

    Codeforces Round #486 (Div. 3) F. Rain and Umbrellas 题目连接: http://codeforces.com/group/T0ITBvoeEx/co ...

  8. D. Frets On Fire 【二分,前缀和】 (Codeforces Global Round 2)

    题目传送门:http://codeforces.com/contest/1119/problem/D D. Frets On Fire time limit per test 1.5 seconds ...

  9. Codeforces Round #436 (Div. 2) E. Fire

    http://codeforces.com/contest/864/problem/E 题意: 有一堆物品,每个物品有3个属性,需要的时间,失效的时间(一开始)和价值.只能一件一件的选择物品(即在选择 ...

随机推荐

  1. tomcat在linux下安装

    1.下载地址: https://tomcat.apache.org/download-90.cgi 2.上传linux 3.查看是否上传成功 4.解压: 5.进入后,查看README.md文件,可以查 ...

  2. 网络端口及nmap扫描

    端口: 计算机与外界交流的出口,在渗透测试当中常用的端口号: 21号端口FTP:文件传输协议 23号端口Telent :远程登录接口 53号端口 DNS: 域名端口 80号端口HTTP:超文本传输协议 ...

  3. 【记】《.net之美》之读书笔记(一) C#语言基础

    前言 工作之中,我们习惯了碰到任务就直接去实现其业务逻辑,但是C#真正的一些基础知识,在我们久而久之不去了解巩固的情况下,就会忽视掉.我深知自己正一步步走向只知用法却不知原理的深渊,所以工作之余,一直 ...

  4. getopt函数用法

    getopt被用来解析命令行选项参数. #include <unistd.h>      extern char *optarg;  //选项的参数指针      extern int o ...

  5. 【题解】[LNOI2014]LCA

    题目戳我 \(\text{Solution:}\) 这题的转化思想很巧妙-- 考虑把\(dep\)给拆掉. 首先可以明确的是,每一个\(LCA\)一定在\(root\to z\)的路径上. 考虑一个\ ...

  6. Redash 二开 - 后端环境搭建

    Redash 二开 - 后端环境搭建 一.操作系统选择 官方文档有一句话:Windows users: while it should be possible to run Redash on a W ...

  7. Zookeeper基础理论

    Zookeeper是分布式开源协调服务, 主要用来解决分布式集群中应用系统的一致性问题. 本质上是分布式小文件存储系统.   特性 全局数据一致性(集群中每个服务器保存一份相同的数据副本,Client ...

  8. 感觉学java学到自己的瓶颈期了,各种框架乱七八糟,感觉好乱。该怎么办!?

    通常我们都会有这样的一个疑问! 解决办法 这时候,你需要的是分清条理,重整知识架构 GitHub开源社区有一个这样的项目,我觉得非常好,很适合Java有基础但是想进阶提升的人. 项目简介 本期介绍的开 ...

  9. VMware安装的Linux系统忘记密码 怎么修改root密码

    因为昨天新安装过虚拟机设置了新的密码,再加上我好长时间没有用自己旧的虚拟机,导致忘记了密码,原来虽然知道在单用模式下,找回密码,但是确实是自己从来都没有做过,还好我们组大手飞翔哥告诉了我,怎么找回ro ...

  10. SQl编程存储过程

    过程化存储 存储过程,一组为完成特定功能.经过编译后存储在数据库中的SQL语序集 灵活性:存储过程中可以进行流程控制和循环操作来完成复杂的判断和运算 一致性:通过存储过程可以使一些关联的操作一起发生, ...