题意:给定平面上一个圆和一堆圆外的点,要求选出尽可能多的点使得它们之间两两连线都不和圆相交。保证任意两点连线不和圆相切。点数<=2000

这题是很久以前在某张课件上看见的。看了题解还搞了三小时,联赛滚粗既视感。

有个结论我是直接看的课件:“点和点连线不与圆相交”对应“这两个点所对的圆上的极角序区间相交但不相互包含”,并不会证,画画图好像是对的。

在直线上选择一些两两相交且不包含的区间[L1,R1],[L2,R2],[L3,R3]….,按左端点排序之后,必然有L1<L2<L3….<Ln<R1<…<Rn,然后课件让我枚举第一个区间是谁,按左端点排序后对右端点跑LIS,O(N^2logN). 这里认为左端点可以在圆上所在区间里逆时针走到所在区间的右端点。

但是我写了一发过不了样例,因为在一个环上选一些两两相交不重合的区间,并不一定存在一个区间把其他区间的左端点都包含起来,比如说把圆三等分成三个区间,再把每个区间稍微左右延长一点点。

懵逼…不过仔细观察,发现这个时候如果选了一个起始区间直接跑LIS,出错时一定是有一个区间左端点不被选中的起始区间包含,但这个区间的右端点绕回了起始区间。那么我们把这个区间变成它在圆上的补区间,好像就挺对的。

所以我们把“左端点不在起始区间但右端点在起始区间内”的区间变成它的补区间,也就是把左右端点交换一下,然后跑LIS就A了。这么做的原理是假如区间A和区间B相交且不包含,那么区间A的补区间也一定和区间B相交且不包含。但是,区间A,B都取补区间后不一定相交且不包含。所以跑完一遍LIS还得把交换过的左右端点恢复回去,实现时我写了一发时间戳标记每个区间是否被交换。

PS. 极角序区间预处理的时候注意所求极角的范围。

/**************************************************************
Problem: 4660
User: liu_runda
Language: C++
Result: Accepted
Time:5540 ms
Memory:944 kb
****************************************************************/ #include<cstdio>
#include<cmath>
#include<algorithm>
#include<cstring>
using namespace std;
const int maxn=2005;
const double pi=3.14159265;
struct point{
double x,y;
void read(){
scanf("%lf %lf",&x,&y);
}
point(){
}
point(double _x,double _y){
x=_x;y=_y;
}
point operator-(const point &B)const{
return point(x-B.x,y-B.y);
}
}P[maxn];
double l[maxn],r[maxn];
double angle(point A){
return atan2(A.y,A.x);
}
int seq[maxn];int tot=0;
bool betw(double l,double r,double x){
if(l<r){
return l<x&&x<r;
}else{
return (x>l)||(x<r);
}
}
bool cmp1(const int &a,const int &b){
return l[a]<l[b];
}
double curl,curr;
bool cmp2(const int &a,const int &b){
if(a>curl&&b>curl)return l[a]<l[b];
else if(a<curr&&b<curr)return l[a]<l[b];
else return l[a]>curl;
}
double m[maxn];
double _r[maxn];
int lis(){
memset(m,0x42,sizeof(m));
m[0]=-1e30;
for(int i=1;i<=tot;++i){
_r[i]=r[seq[i]]-curl;
if(_r[i]<0)_r[i]+=2*pi;
}/*
for(int i=1;i<=tot;++i)printf("%.2f ",_r[i]);
printf("\n");
for(int i=1;i<=tot;++i){
printf("%.2f %.2f\n",l[seq[i]],r[seq[i]]);
}*/ int ans=0;double *pt;
for(int i=1;i<=tot;++i){
pt=upper_bound(m,m+tot+1,_r[i]);
if((pt-m)>ans)ans=pt-m;
*pt=_r[i];
}
return ans;
}
int T;
int mark[maxn];
int main(){
// freopen("4660.in","r",stdin);
// freopen("4660.out","w",stdout);
int n,R;
scanf("%d%d",&n,&R);
for(int i=1;i<=n;++i)P[i].read();
double tmp,delta;
for(int i=1;i<=n;++i){
tmp=angle(P[i]);delta=fabs(acos(R/sqrt(P[i].x*P[i].x+P[i].y*P[i].y)));
// printf("%.2f %.2f\n",tmp,delta);
l[i]=tmp-delta;r[i]=tmp+delta;
if(l[i]<-pi)l[i]+=pi*2;
if(r[i]>pi)r[i]-=pi*2;
}//printf("\n");
/* for(int i=1;i<=n;++i){
printf("%.2f %.2f\n",l[i],r[i]);
}//while(1);
printf("\n");*/
int ans=0,curans;
for(int i=1;i<=n;++i){//枚举起点
tot=0;
++T;
seq[++tot]=i;
for(int j=1;j<=n;++j){
if(i!=j&&betw(l[i],r[i],l[j])&&!betw(l[i],r[i],r[j]))seq[++tot]=j;
else if(i!=j&&betw(l[i],r[i],r[j])&&!betw(l[i],r[i],l[j])){
swap(l[j],r[j]);seq[++tot]=j;
mark[j]=T;
}
}
curl=l[i];curr=r[i];
if(l[i]<r[i])sort(seq+1,seq+tot+1,cmp1);
else sort(seq+1,seq+tot+1,cmp2);
curans=lis();
if(curans>ans)ans=curans;
for(int j=1;j<=tot;++j){
if(mark[j]==T)swap(l[j],r[j]);
}
}
printf("%d\n",ans);//while(1);
return 0;
} /*
5 8
-7 20
10 80
-7 26
-13 79
11 81
*/

  

Bzoj3663/4660 CrazyRabbit的更多相关文章

  1. 三倍经验——bzoj3663、4660、4206 Crazy Rabbit/最大团

    题目描述: 3663 4660 4206 题解: 第一眼:不成立的互相连边,然后用网络流求解无向图最小点覆盖! 好吧我不会. 正解: 每个点对应圆上的一段圆弧,长这样: 设对应圆弧$(l,r)$. 若 ...

  2. bzoj 4660 Crazy Rabbit——LIS解决“相交”限制的思想

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4660 想到一个点可以用它与圆的两个切点表示.并想到可以把切点极角排序,那么就变成环上的一些区 ...

  3. bzoj3663/4660CrazyRabbit && bzoj4206最大团

    题意 给出平面上N个点的坐标,和一个半径为R的圆心在原点的圆.对于两个点,它们之间有连边,当且仅当它们的连线与圆不相交.求此图的最大团. 点数<=2000,坐标的绝对值和半径<=5000. ...

  4. bzoj 3768: spoj 4660 Binary palindrome二进制回文串

    Description 给定k个长度不超过L的01串,求有多少长度为n的01串S满足: 1.该串是回文串 2.该串不存在两个不重叠的子串,在给定的k个串中. 即不存在a<=b<c<= ...

  5. 2018.10.30 一题 洛谷4660/bzoj1168 [BalticOI 2008]手套——思路!问题转化与抽象!+单调栈

    题目:https://www.luogu.org/problemnew/show/P4660 https://www.lydsy.com/JudgeOnline/problem.php?id=1168 ...

  6. bzoj3663

    几何+lis 很巧妙.直接做很困难,那么我们转化一下,把每个点能看见的圆弧画出来.只有这些圆弧相交时才满足条件. 那么也就是找出圆上尽量多两两相交的区间. 所以我们先按左端点极角排序,然后固定一个必须 ...

  7. [Keras] Develop Neural Network With Keras Step-By-Step

    简单地训练一个四层全连接网络. Ref: http://machinelearningmastery.com/tutorial-first-neural-network-python-keras/ 1 ...

  8. EChart数据的异步加载和更新

    ECharts是国内开发一款图标插件,在网页中我们经常要用到图标显示,直接引用十分方便. 直接到ECharts主页调用插件 <!DOCTYPE html> <html style=& ...

  9. 通用js地址选择器

    用js实现通用的地址选择器,省份,城市,地区自动关联更新 点击下面查看详细代码: http://runjs.cn/code/s8sqkhcv 关键地址库代码: var addr_arr = new A ...

随机推荐

  1. 清北学堂2017NOIP冬令营入学测试P4745 B’s problem(b)

    清北学堂2017NOIP冬令营入学测试 P4745 B's problem(b) 时间: 1000ms / 空间: 655360KiB / Java类名: Main 背景 冬令营入学测试 描述 题目描 ...

  2. CAP原理的证明

    CAP概述 C: Consistency 一致性 A: Availability 可用性 P:Partition Tolerance分区容错性 CAP理论的核心是:一个分布式系统不可能同时很好的满足一 ...

  3. PRML读书会第十章 Approximate Inference(近似推断,变分推断,KL散度,平均场, Mean Field )

    主讲人 戴玮 (新浪微博: @戴玮_CASIA) Wilbur_中博(1954123) 20:02:04 我们在前面看到,概率推断的核心任务就是计算某分布下的某个函数的期望.或者计算边缘概率分布.条件 ...

  4. Canvas之打字机游戏

    最近针对粒子化作了一点点的探究,决定结合其做个小游戏,于是这个简单的打字游戏出世了. 试玩地址:Typewriter game  仅在chrome下测试,请谨慎使用其他浏览器(特别是ff):加载速度有 ...

  5. 我的职业生涯总结---班门弄斧之我们该怎样从零开始学习.NET

    标题说的很清楚了,这篇文章纯属班门弄斧,大神可随意喷.我只是结合自己4年不到的学习与使用.NET的水平. 首先说下这篇博客的背景吧.前两天有个我的读者加我微信,然后就有了下面这样的对话, 可能有些人第 ...

  6. unity3d Vector3.Lerp解析

    Vector3.Lerp:http://www.ceeger.com/Script/Vector3/Vector3.Lerp.html 手册中描述的不是很详细,什么叫“按照数字t在from到to之间插 ...

  7. js异步状态监控

    说明:写这篇文章,是希望被吐槽的. 一.背景 在做报表页面的时候,页面上有很多的异步加载,而设计的loading是个全局的,一个页面就有一个. 控制loading什么时候出现,什么时候消失,要实时的知 ...

  8. lambda函数、lambda表达式

    C++11 新特性:Lambda 表达式 豆子 2012年5月15日 C++ 10条评论 参考文章:https://blogs.oracle.com/pcarlini/entry/c_1x_tidbi ...

  9. A Regularized Competition Model for Question Diffi culty Estimation in Community Question Answering Services-20160520

    1.Information publication:EMNLP 2014 author:Jing Liu(在前一篇sigir基础上,拓展模型的论文) 2.What 衡量CQA中问题的困难程度,提出从两 ...

  10. WPS显示无法创建对象,请确认对象已在系统注册表中注册

    第一种方法:在系统的开始--所有程序找到WPS--WPS office工具--配置工具--高级--兼容设置,在这里勾选兼容第三方系统和软件. 第二种方法: xp/win7系统:拷贝packager.e ...