CodeForces 1419F Rain of Fire
题意
不想写。
题解
场上想了 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的更多相关文章
- Codeforces 988F Rain and Umbrellas(DP)
题目链接:http://codeforces.com/contest/988/problem/F 题目大意: 有三个整数a,n,m,a是终点坐标,给出n个范围(l,r)表示这块区域下雨,m把伞(p,w ...
- Codeforces Round #436 E. Fire(背包dp+输出路径)
题意:失火了,有n个物品,每个物品有价值pi,必须在时间di前(小于di)被救,否则就要被烧毁.救某个物 品需要时间ti,问最多救回多少价值的物品,并输出救物品的顺序. Examples Input ...
- Codeforces 988F. Rain and Umbrellas
解题思路:动态规划 遍历点i,如果从前一个点i-1走到这个点i不需要伞,则疲劳值不变dp[i] = dp[i-1]. 如果前一个点i-1走到这一个点i需要伞,则从前面找一把伞. 即遍历前面的每个点j, ...
- CodeForces 题目乱做
是个补题记录. 1419 除了 F 场上都过了. CF1419A Digit Game 这题好多人 FST 啊-- 考虑如果串长为奇数那么最后操作的肯定是第一个人,串长为偶数的最后操作的肯定是第二个, ...
- little alchemy攻略
一个造物游戏: acidrain=rain+smoke airlplain=metal+bird alcohol=fruit+time algae=plant+water allergy=dust+h ...
- [Codeforces 864E]Fire
Description Polycarp is in really serious trouble — his house is on fire! It's time to save the most ...
- Codeforces Round #486 (Div. 3) F. Rain and Umbrellas
Codeforces Round #486 (Div. 3) F. Rain and Umbrellas 题目连接: http://codeforces.com/group/T0ITBvoeEx/co ...
- 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 ...
- Codeforces Round #436 (Div. 2) E. Fire
http://codeforces.com/contest/864/problem/E 题意: 有一堆物品,每个物品有3个属性,需要的时间,失效的时间(一开始)和价值.只能一件一件的选择物品(即在选择 ...
随机推荐
- 创建Vue项目及封装axios
1. 始vue化项目 https://www.cnblogs.com/xiaonq/p/11027880.html vue init webpack deaxios # 使用脚手架创建项目 deaxi ...
- Python 自动化测试全攻略:五种自动化测试模型实战详解
随着移动互联网的发展,软件研发模型逐步完善,软件交付质量越来越受到软件公司的重视,软件测试技术特别是自动化测试技术开始在软件系统研发过程中发挥着越来越重要的作用. 与传统的手工测试技术相比,自动化测试 ...
- 基于空镜像scratch创建一个新的Docker镜像
我们在使用Dockerfile构建docker镜像时,一种方式是使用官方预先配置好的容器镜像.优点是我们不用从头开始构建,节省了很多工作量,但付出的代价是需要下载很大的镜像包. 比如我机器上docke ...
- 一些IT service的相关知识
1. cmd是什么,怎么在电脑上打开cmd命令框. 在windows环境下,命令行程序为cmd.exe,是一个32位的命令行程序,微软Windows系统基于Windows上的命令解释程序,类似于微软的 ...
- Prometheus第一篇:Prometheus架构解析
Prometheus是新一代的监控系统解决方案,原生支持云环境,和kubernetes无缝对接,的却是容器化监控解决方案的不二之选.当然对传统的监控方案也能够兼容,通过自定义或是用开源社区提供的各种e ...
- Fabric1.4.4 基础环境搭建
简单记录一下fabric版本1.4.4的环境搭建部署,运行环境为CentOs7.8,如有错误欢迎批评指正. 1.Docker 和 Docker Compose 1. docker的安装部署 docke ...
- arcgis-java-100.8.0.jar下载
链接: https://pan.baidu.com/s/1HoW2IhPvHRw9LBZphxC5Rw 提取码: pexn
- UDP协议网络Socket编程(java实现C/S通信案例)
我的博客园:https://www.cnblogs.com/chenzhenhong/p/13825286.html 我的CSDN博客:https://blog.csdn.net/Charzous/a ...
- 【换根DP】小奇的仓库
题目背景 小奇采的矿实在太多了,它准备在喵星系建个矿石仓库.令它无语的是,喵星系的货运飞船引擎还停留在上元时代! 题目内容 喵星系有\(n\)个星球,星球以及星球间的航线形成一棵树. 从星球\(a\) ...
- rabbitmq 延时队列 插件方式实现 每条消息都延时自己时间
上篇文章的延时是加到队列上的 通过死信过时推送 ,缺点就是不能每条消息定义自己的过时时间而且每次有新的过时时间,要新建一个交换机和队列 https://www.cnblogs.com/brady-wa ...