洛谷P4518 [JSOI2018]绝地反击(计算几何+二分图+退流)
题面
题解
调了咱一个上午……
首先考虑二分答案,那么每个点能够到达的范围是一个圆,这个圆与目标圆的交就是可行的区间,这个区间可以用极角来表示
首先,如果我们知道这个正\(n\)边形的转角,也就是它在水平的基础上转过了几度的话,那么可以把它的每个顶点和包含它的圆弧所代表的点连边,如果这个二分图存在完备匹配那么说明有解
然而我们并不知道这个多边形转过了几度
我们考虑一种可行的方案,如果它没有任何一个顶点和在一段圆弧的端点上,那么一定可以转一点点距离使其中一个顶点刚好落在一个圆弧的端点上,那么我们要考虑的转角实际上只有\(2n\)种
发现这是个正多边形,所以我们把所有的转角对\({2\pi\over n}\)取模,那么是等价的。然后把所有的转角从小到大排序,那么转角每次增大的时候只会新增一条边或者减少一条边。那么我们就不需要重新构图了,删边的时候直接退流,加边的时候看一下有没有新的增广路就行了
//minamoto
#include<bits/stdc++.h>
#define R register
#define inf 0x3f3f3f3f
#define fp(i,a,b) for(R int i=(a),I=(b)+1;i<I;++i)
#define fd(i,a,b) for(R int i=(a),I=(b)-1;i>I;--i)
#define go(u) for(int i=head[u],v=e[i].v;i;i=e[i].nx,v=e[i].v)
#define gg(u) for(int &i=cur[u],v=e[i].v;i;i=e[i].nx,v=e[i].v)
template<class T>inline bool cmin(T&a,const T&b){return a>b?a=b,1:0;}
template<class T>inline bool cmax(T&a,const T&b){return a<b?a=b,1:0;}
using namespace std;
char buf[1<<21],*p1=buf,*p2=buf;
inline char getc(){return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++;}
int read(){
R int res,f=1;R char ch;
while((ch=getc())>'9'||ch<'0')(ch=='-')&&(f=-1);
for(res=ch-'0';(ch=getc())>='0'&&ch<='9';res=res*10+ch-'0');
return res*f;
}
const int N=605,M=N*N;const double Pi=acos(-1.0),eps=1e-8;
struct eg{int v,nx,w;}e[M];int head[N],tot;
inline void add(R int u,R int v,R int w){
e[++tot]={v,head[u],w},head[u]=tot;
e[++tot]={u,head[v],0},head[v]=tot;
}
int n,r,S,T,h,t,u,v,x,y,flow,cur[N],dep[N],q[N];
bool bfs(){
fp(i,S,T)cur[i]=head[i],dep[i]=-1;
q[h=t=1]=S,dep[S]=0;
while(h<=t){
u=q[h++];
go(u)if(dep[v]<0&&e[i].w){
dep[v]=dep[u]+1,q[++t]=v;
if(v==T)return true;
}
}
return false;
}
int dfs(int u,int lim){
if(u==T||!lim)return lim;
int fl=0,f;
gg(u)if(dep[v]==dep[u]+1&&(f=dfs(v,min(lim,e[i].w)))){
fl+=f,lim-=f,e[i].w-=f,e[i^1].w+=f;
if(!lim)break;
}if(!fl)dep[u]=-1;
return fl;
}
void del(int u,int v){
for(R int i=head[v],j=0;i;j=i,i=e[i].nx)
if(e[i].v==u){j?(e[j].nx=e[i].nx):(head[v]=e[i].nx);break;}
for(R int i=head[u],j=0;i;j=i,i=e[i].nx)
if(e[i].v==v){
j?(e[j].nx=e[i].nx):(head[u]=e[i].nx);
if(e[i].w)return;
break;
}
--flow;
for(int i=head[S];i;i=e[i].nx)if(e[i].v==u){e[i].w^=1,e[i^1].w^=1;break;}
for(int i=head[T];i;i=e[i].nx)if(e[i].v==v){e[i].w^=1,e[i^1].w^=1;break;}
if(bfs())flow+=dfs(S,inf);
}
struct point{
int x,y;double len;
point(){}
point(R int xx,R int yy):x(xx),y(yy){len=sqrt(x*x+y*y);}
}p[N];
struct node{
double t;int u,v,op;
node(){}
node(R double tt,R int uu,R int vv,R int oop):t(tt),u(uu),v(vv),op(oop){}
inline bool operator <(const node &b)const{return t==b.t?op>b.op:t<b.t;}
}st[N];
double alp,ql,qr,mid;
bool ck(double mid){
fp(i,S,T)head[i]=0;tot=1;
int top=0;
fp(i,1,n){
if(p[i].len+r<=mid)fp(j,1,n)add(i,j+n,1);
else{
R double t=acos((p[i].len*p[i].len+r*r-mid*mid)/(2*p[i].len*r));
R double base=atan2(p[i].y,p[i].x);
R double l=base-t,r=base+t;
l<0?l+=2*Pi:0,r<0?r+=2*Pi:0;
int ll=l/alp,rr=r/alp;
st[++top]=node(l-ll*alp,i,ll+1,1),
st[++top]=node(r-rr*alp,i,rr+1,-1);
++ll,++rr;
if(l<=r)fp(j,ll+1,rr)add(i,j+n,1);
else{
fp(j,ll+1,n)add(i,j+n,1);
fp(j,1,rr)add(i,j+n,1);
}
}
}
fp(i,1,n)add(S,i,1),add(i+n,T,1);
sort(st+1,st+1+top),flow=0;
while(bfs())flow+=dfs(S,inf);
if(flow==n)return true;
fp(i,1,top)if(st[i].op<0)del(st[i].u,st[i].v+n);
else{
add(st[i].u,st[i].v+n,1);
if(bfs())flow+=dfs(S,inf);
if(flow==n)return true;
}
return false;
}
int main(){
// freopen("testdata.in","r",stdin);
n=read(),r=read(),alp=2*Pi/n;
S=0,T=n+n+1;
fp(i,1,n)x=read(),y=read(),p[i]=point(x,y),cmax(ql,fabs(p[i].len-r)),cmax(qr,p[i].len+r);
while(qr-ql>eps)ck(mid=(ql+qr)/2)?qr=mid:ql=mid;
printf("%.8lf\n",ql);
return 0;
}
洛谷P4518 [JSOI2018]绝地反击(计算几何+二分图+退流)的更多相关文章
- 洛谷P4014 分配问题【最小/大费用流】题解+AC代码
洛谷P4014 分配问题[最小/大费用流]题解+AC代码 题目描述 有 n 件工作要分配给 n 个人做.第 i 个人做第 j 件工作产生的效益为c ij. 试设计一个将 n 件工作分配给 n 个人做的 ...
- 洛谷P4559 [JSOI2018]列队 【70分二分 + 主席树】
题目链接 洛谷P4559 题解 只会做\(70\)分的\(O(nlog^2n)\) 如果本来就在区间内的人是不用动的,区间右边的人往区间最右的那些空位跑,区间左边的人往区间最左的那些空位跑 找到这些空 ...
- 洛谷 P4016负载平衡问题【费用流】题解+AC代码
洛谷 P4016负载平衡问题 P4014 分配问题[费用流]题解+AC代码 负载平衡问题 题目描述 GG 公司有n个沿铁路运输线环形排列的仓库,每个仓库存储的货物数量不等.如何用最少搬运量可以使 n ...
- 洛谷 P4012 深海机器人问题【费用流】
题目链接:https://www.luogu.org/problemnew/show/P4012 洛谷 P4012 深海机器人问题 输入输出样例 输入样例#1: 1 1 2 2 1 2 3 4 5 6 ...
- 【洛谷2053】 [SCOI2007]修车(费用流)
传送门 洛谷 Solution 考虑把每一个修车工人拆成\(n\)个点,那么考虑令\(id(i,j)\)为第\(i\)个工人倒数第\(j\)次修车. 然后就可以直接跑费用流了!!! 代码实现 /* m ...
- 【洛谷2050】 [NOI2012]美食节(费用流)
大家可以先看这道题目再做! SCOI2007修车 传送门 洛谷 Solution 就和上面那道题目一样的套路,但是发现你会获得60~80分的好成绩!!! 考虑优化,因为是SPFA,所以每一次只会走最短 ...
- 洛谷P3376【模板】网络最大流 ISAP
这篇博客写得非常好呀. 传送门 于是我是DCOI这一届第一个网络流写ISAP的人了,之后不用再被YKK她们嘲笑我用Dinic了!就是这样! 感觉ISAP是会比Dinic快,只分一次层,然后不能增广了再 ...
- 洛谷P1330封锁阳光大学[二分图染色]
题目描述 曹是一只爱刷街的老曹,暑假期间,他每天都欢快地在阳光大学的校园里刷街.河蟹看到欢快的曹,感到不爽.河蟹决定封锁阳光大学,不让曹刷街. 阳光大学的校园是一张由N个点构成的无向图,N个点之间由M ...
- 洛谷 P3386 【模板】二分图匹配
题目背景 二分图 题目描述 给定一个二分图,结点个数分别为n,m,边数为e,求二分图最大匹配数 输入输出格式 输入格式: 第一行,n,m,e 第二至e+1行,每行两个正整数u,v,表示u,v有一条连边 ...
随机推荐
- 委托BegionInvoke和窗体BegionInvoke
委托BegionInvoke是指通过委托方法执行多线程任务,例如: //定义委托成员变量 delegate void dg_DeleAirport(); //指定委托函数 dg_DeleAirpor ...
- c#winform图片绘制与图片验证码
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...
- 使用clr 调用C#编写的dll中的方法的全解释
使用clr 调用C#编写的dll中的方法的全解释1.数据库初始化:将下面这段代码直接在运行就可以初始化数据库了exec sp_configure 'show advanced options', '1 ...
- 58. Length of Last Word最后一个单词的长度
[抄题]: [暴力解法]: 时间分析: 空间分析: [优化后]: 时间分析: 空间分析: [奇葩输出条件]: [奇葩corner case]: "b a " 最后一位是空格,可能误 ...
- linux下配置eclipse环境
注明:本文为博主原创文章,转载请注明出处 前期准备 (此文使用的是非安装版jdk1.8,你也可以下载版本更低的,而且建议使用1.6版本,66大顺嘛,嘻嘻) 1.点击下载jdk 2.点击下载eclips ...
- 数据库MySQL之 视图、触发器、存储过程、函数、事务、数据库锁、数据库备份、事件
数据库MySQL之 视图.触发器.存储过程.函数.事务.数据库锁.数据库备份.事件 浏览目录 视图 触发器 存储过程 函数 事务 数据库锁 数据库备份 事件 一.视图 1.视图概念 视图是一个虚拟表, ...
- [GO]数组的比较和赋值
package main import "fmt" func main() { //支持比较,只支持==或!=,比较是不是每一个元素都一样,2个数据比较,数据类型要一样 a := ...
- HYSBZ 1036 树的统计Count (水题树链剖分)
题意:中文题. 析:就是直接维护一个最大值和一个和,用线段树维护即可,这个题很简单,但是我卡了一晚上,就是在定位的时候,位置直接反过来了,但是样例全过了...真是... 代码如下: #pragma c ...
- 设计模式08: Composite 组合模式(结构型模式)
Composite 组合模式(结构型模式) 对象容器的问题在面向对象系统中,我们常会遇到一类具有“容器”特征的对象——即他们在充当对象的同时,又是其他对象的容器. public interface I ...
- [转]windows7远程桌面连接失败:发生身份验证错误。要求的函数不受支持
转至:https://jingyan.baidu.com/article/d169e18604ca86436611d821.html 系统升级后出现远程连接报错,“发生身份验证错误.要求的函数不受支持 ...