[Contest20180321]nonintersect

$\dfrac 2\pi$是个引人注目的数字,先来看看它到底是什么东西
假如有一条直线,它和题目所给的某条长度为$d$的线段夹角为$\theta$,那么线段在直线上的投影长度为$\left|d\cos\theta\right|$
现在我们随机一条直线,于是线段的期望投影长度是$\dfrac{\int_0^\pi\left|d\cos\theta\right|d\theta}{\pi-0}=\dfrac2\pi d$
所以我们可以不停随机一个角度,把所有线段投影到这个角度的直线上,一旦检查到总投影长度与原长之比$\geq\dfrac2\pi$就停止
找(?)到合适的角度之后,我们按线段端点投影在直线上的横坐标把所有点分成左右两部分,找一种连线方法使得每条线端的两个端点一个在左边一个在右边,且连线不相交,这样就做完了这道题
当然一些细节是不得不讲的
①随机?问题不大...期望都是$\dfrac2\pi$了,想随机到一个比它大的一点都不难(这也顺带说明了不存在无解的情况)
②设原来线段总长为$len$,原来投影总长为$d$,求得答案的投影总长为$d'$,求得答案的线段总长为$len'$,那么显然有$\dfrac2\pi len\leq d$和$d'\leq len'$
考虑比较$d$和$d'$,我们要求答案的所有线段的端点分列左右两边,对应过来就是投影线段穿越中轴线,如果原来存在两条投影线段不穿越中轴线,我们换一种方式连接即可满足要求,这样一来投影长度还增加了,所以$d\leq d'$

③怎么用合适的方法连线(不相交)
我们可以这样做:每次选取左半边的最左最下点,把其他点做极角排序,扫描的过程中统计扫描线下方左右两边的点数,一旦相同,连一条边并递归上下处理,这样就保证了连线不相交

实现的时候可以不用写成递归的形式,对于分开的上下两组,分别打上不同的标记即可,下次处理到这里只需要找标记相同的点,跟递归差不多的意思gr
然后就做完了,挺愉悦的==
#include<stdio.h>
#include<math.h>
#include<algorithm>
using namespace std;
const double pi=3.141592653589793238462643383;
struct point{
double x,y;
point(double a=0,double b=0){x=a;y=b;}
}p[10010],a[10010];
double dis(point a,point b){return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y));}
int match[10010],id[10010],s[10010],c[10010],bl[10010];
bool v[10010];
double ang[10010];
bool cmpx(int x,int y){return a[x].x<a[y].x||(a[x].x==a[y].x&&a[x].y<a[y].y);}
bool cmpa(int x,int y){return ang[x]<ang[y];}
int main(){
int n,i,j,k,x,y,l,c0,c1,tot;
double sum,tmp,d,dx,dy;
scanf("%d",&n);
for(i=1;i<=n<<1;i++)scanf("%lf%lf",&p[i].x,&p[i].y);
sum=0;
for(i=1;i<=n;i++){
scanf("%d%d",&x,&y);
sum+=dis(p[x],p[y]);
}
while(1){
d=(rand()%10000)/10000.*2*pi;
dx=cos(d);
dy=sin(d);
for(i=1;i<=n<<1;i++)a[i]=point(p[i].x*dx-p[i].y*dy,p[i].x*dy+p[i].y*dx);
for(i=1;i<=n<<1;i++)id[i]=i;
sort(id+1,id+(n<<1|1),cmpx);
tmp=0;
for(i=1;i<=n;i++)tmp+=a[id[i+n]].x-a[id[i]].x;
if(tmp>=2/pi*sum)break;
}
for(i=1;i<=n;i++){
s[id[i]]=0;
s[id[i+n]]=1;
}
tot=0;
for(i=1;i<=n;i++){
x=id[i];
v[x]=1;
l=0;
for(j=1;j<=n<<1;j++){
if(!v[j]&&bl[j]==bl[x]){
ang[j]=atan2(a[j].y-a[x].y,a[j].x-a[x].x);
l++;
c[l]=j;
}
}
sort(c+1,c+l+1,cmpa);
c0=c1=0;
for(j=1;j<=l;j++){
y=c[j];
if(s[x]!=s[y]&&c0==c1){
match[x]=y;
match[y]=x;
v[y]=1;
tot++;
for(k=1;k<j;k++)bl[c[k]]=tot;
break;
}
if(s[y])
c1++;
else
c0++;
}
}
for(i=1;i<=n<<1;i++){
if(match[i]>i)printf("%d %d\n",i,match[i]);
}
}
[Contest20180321]nonintersect的更多相关文章
- 【XSY2760】nonintersect 计算几何
题目描述 平面上有\(n\)条线段,你要擦掉所有线段但保留原有的\(2n\)个端点,然后连接这些端点形成\(n\)条不相交的线段,每个端点只能在一条线段中. 假设你画的线段总长为\(Y\),原有线段的 ...
- [算法]检测空间三角形相交算法(Devillers & Guigue算法)
#pragma once //GYDevillersTriangle.h /* 快速检测空间三角形相交算法的代码实现(Devillers & Guigue算法) 博客原地址:http://bl ...
- Foundations of Machine Learning: The PAC Learning Framework(1)
写在最前:本系列主要是在阅读 Mehryar Mohri 等的最新书籍<Foundations of Machine Learning>以及 Schapire 和 Freund 的 < ...
随机推荐
- [poj 3436]最大流+输出结果每条边流量
题目链接:http://poj.org/problem?id=3436 大力套kuangbin板过了orz #include<cstdio> #include<cstring> ...
- grub ubuntu启动
set root=(hd0,gpt10) 现在变为 gpt9 了 安装固态后.变成了 (hd1,gpt11) set prefix=(hd0,gpt10)/boot/grub insmod norma ...
- pycharm激活(JetBrains IDEA 系列产品通用xx方法(license server))
http://xclient.info/a/f0b9738a-36fd-8a97-a966-0d3db497092d.html .打开激活窗口 .选择 Activate new license wit ...
- eclipse关闭错误警告提示
- centos 安装mysql 笔记
1.查询已安装软件的目录 rpm -ql mysql 2.mysql的安装卸载 a. 查找已安装的myslq 版本: #rpm -qa | grep mysql (注意大小写,如果mysql 不行 ...
- Java并发(6)- CountDownLatch、Semaphore与AQS
引言 上一篇文章中详细分析了基于AQS的ReentrantLock原理,ReentrantLock通过AQS中的state变量0和1之间的转换代表了独占锁.那么可以思考一下,当state变量大于1时代 ...
- .NET的PE文件结构篇(转)
一.开篇 开篇我要讲述一个关于PE文件结构的文章,这篇文章动手能力比较强,希望大家能够动手进行操作,这边文章篇幅有可能会长一些,为了方便大家阅读我可以将其分为几个部分进行讲解,主要分为以下几个部分: ...
- 转:Spring AOP详解
转:Spring AOP详解 一.前言 在以前的项目中,很少去关注spring aop的具体实现与理论,只是简单了解了一下什么是aop具体怎么用,看到了一篇博文写得还不错,就转载来学习一下,博文地址: ...
- io缓冲为何可以提高效率
问题 据我了解,运用FileInputStream读写一段数据是一个字节一个字节的读取,如果有10个字节大小的文件,就要调用10次系统调用,每次将读取的数据赋值给变量,然后程序使用变量. 缓冲区可以看 ...
- html初探
HTML HTML是英文Hyper Text Mark-up Language(超文本标记语言)的缩写,他是一种制作万维网页面标准语言(标记).相当于定义统一的一套规则,大家都来遵守他,这样就可以让浏 ...