[COCI2009]Dvapravca

LG传送门

先给出考场上的\(O(n^3)\)乱搞方法:枚举一个蓝点和一个红点,找出过着两个点的直线,再枚举蓝点找出这条直线最多能往两边扩展多宽,最后枚举红点计算贡献。

注意在确定一条直线能往两边扩展多宽时不要求点到直线的距离,否则常数会太大,只要求竖直方向的距离就可以了。正确性显然,具体看代码。

现在有个重要的问题,求大佬来解决:如果直接这样写会被卡到65到70分,但是如果把蓝点和红点按\(x\)为第一关键字\(y\)为第二关键字从小到大排个序,就能得到90至100分(考试时用的评测机能跑到90分,洛谷神机上能切)。然而我不会证明这样究竟优化在哪里,求大佬证明。

#include<cstdio>
#include<algorithm>
#define R register
#define I inline
#define D double
using namespace std;
const int N=1003,inf=0x3f3f3f3f;
struct V{int x,y;}r[N],b[N];
struct L{D k,b;}f;
I int operator<(V a,V b){return a.x^b.x?a.x<b.x:a.y<b.y;}
I D min(D x,D y){return x<y?x:y;}
I int max(int x,int y){return x>y?x:y;}
I L lin(V a,V b){
if(a.x==b.x)
return (L){inf,0};
D k=(D)(b.y-a.y)/(b.x-a.x);
return (L){k,(D)a.y-k*a.x};
}
I D dst(V a){return f.k*a.x+f.b-a.y;}
int main(){
R int n,x,y,i,j,t,p=0,q=0,g,h,o=0;
R char c[2];
D d,u,v;
scanf("%d",&n);
for(i=1;i<=n;++i){
scanf("%d%d%s",&x,&y,c);
if(c[0]=='R')
r[++p]=(V){x,y};
else
b[++q]=(V){x,y};
}
sort(b+1,b+q+1),sort(r+1,r+p+1);//就是这里,不加会慢很多
for(i=1;i<=q;++i)
for(j=1;j<=p;++j){
f=lin(b[i],r[j]),u=v=inf,g=h=1;
for(t=1;t<=q;++t)
if(t^i)
if((d=dst(b[t]))>0)
u=min(u,d);
else
v=min(v,-d);
for(t=1;t<=p;++t)
if(t^j)
if((d=dst(r[t]))>0)
g+=d<u;
else
h+=-d<v;
o=max(o,max(g,h));
}
printf("%d",o);
return 0;
}

亲测如果没加sort应该是过不了的,xzz大神仙尝试把坐标系随机旋转了一下,变得更快了。所以这道题告诉我们一个经验:碰见不会的计算几何题,先随机旋转坐标系,再把点排一遍序,打起暴力更有自信!

还是放一下正解\(O(n^2 log n)\)的做法吧。考虑如果要求直线垂直于\(x\)轴,那么如果把点按刚才所说的排一遍序,要求的就是序列上连续的最长的\(1\)的个数(如果把红色视作\(1\))。由于直线可以倾斜,考虑旋转坐标系,当某两个点连线的斜率与当前\(y\)轴在原坐标系中的斜率相等时,这两个点在序列上的位置就会交换,这下可以用线段树来维护,由于最多交换\(O(n^2)\)次顺序,复杂度是\(O(n^2 log n)\)。

#include<cstdio>
#include<algorithm>
#define R register
#define I inline
#define D double
using namespace std;
const int N=1003,M=1000003,S=4003;
int f[N],n;
struct T{int f,l,r,d;}e[S];
struct V{int x,y,c;}p[N];
struct L{D k; int x,y;}q[M];
I int operator<(V a,V b){return a.x^b.x?a.x<b.x:a.y<b.y;}
I int operator<(L a,L b){return a.k>b.k;}
I int max(int x,int y){return x>y?x:y;}
I T operator+(T x,T y){
T z;
z.f=max(x.r+y.l,max(x.f,y.f)),z.d=x.d+y.d;
if(x.l==x.d)
z.l=x.d+y.l;
else
z.l=x.l;
if(y.r==y.d)
z.r=x.r+y.d;
else
z.r=y.r;
return z;
}
I void swp(int&x,int&y){x^=y,y^=x,x^=y;}
I void upd(int k,int v){e[k]=(T){v,v,v,1};}
I void bld(int k,int l,int r){
if(l==r){
upd(k,p[l].c);
return ;
}
R int p=k<<1,q=p|1,m=l+r>>1;
bld(p,l,m),bld(q,m+1,r),e[k]=e[p]+e[q];
}
void mdf(int k,int l,int r,int x,int v){
if(l==r){
upd(k,v);
return ;
}
R int p=k<<1,q=p|1,m=l+r>>1;
if(x<=m)
mdf(p,l,m,x,v);
else
mdf(q,m+1,r,x,v);
e[k]=e[p]+e[q];
}
int main(){
R int i,j,x,y,t=0,o;
R char c[2];
scanf("%d",&n);
for(i=1;i<=n;++i){
scanf("%d%d%s",&x,&y,c);
if(c[0]=='R')
p[i]=(V){x,y,1};
else
p[i]=(V){x,y,0};
}
sort(p+1,p+n+1);
for(i=1;i<=n;++i)
f[i]=i;
for(i=1;i<n;++i)
for(j=i+1;j<=n;++j)
q[++t]=(L){(D)(p[j].y-p[i].y)/(p[j].x-p[i].x),i,j};
sort(q+1,q+t+1),bld(1,1,n),o=e[1].f;
for(i=1;i<=t;++i){
x=f[q[i].x],y=f[q[i].y],swp(f[q[i].x],f[q[i].y]);
if(p[x].c^p[y].c)
swp(p[x].c,p[y].c),mdf(1,1,n,x,p[x].c),mdf(1,1,n,y,p[y].c),o=max(o,e[1].f);
}
printf("%d",o);
return 0;
}

[COCI2009]Dvapravca 计算几何的更多相关文章

  1. bzoj3086: Coci2009 dvapravca

    Description 给定平面上的 N 个点, 其中有一些是红的, 其他是蓝的.现在让你找两条平行的直线, 使得在保证    不存在一个蓝色的点 被夹在两条平行线之间,不经过任何一个点, 不管是蓝色 ...

  2. luogu P4385 [COCI2009]Dvapravca

    传送门 我真的弱,正解都不会还打了个错的暴力 考虑平行线与x轴平行,那么可以按照y为第一关键字升序,x为第二关键字升序排序,然后合法的一段红点就是连续的一段,答案也就是最大的连续红色段 推广到一般情况 ...

  3. P4385 [COCI2009]Dvapravca

    首先特判掉蓝点数量\(<2\)的情况.没有蓝点答案就是\(n\),有一个蓝点可以枚举一个红点,选择过这个蓝点和红点的一条线和在无穷远处的平行线(即这条线对应的两个半平面). 这里认为过一个点是与 ...

  4. [洛谷P4385][COCI2009]Dvapravca(咕咕咕)

    题目大意:很早以前做的题 题解: 卡点: C++ Code: #pragma GCC optimize("Ofast") #pragma GCC optimize("un ...

  5. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  6. ACM/ICPC 之 计算几何入门-叉积-to left test(POJ2318-POJ2398)

    POJ2318 本题需要运用to left test不断判断点处于哪个分区,并统计分区的点个数(保证点不在边界和界外),用来做叉积入门题很合适 //计算几何-叉积入门题 //Time:157Ms Me ...

  7. HDU 2202 计算几何

    最大三角形 Time Limit: 5000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submi ...

  8. ACM 计算几何中的精度问题(转)

    http://www.cnblogs.com/acsmile/archive/2011/05/09/2040918.html 计算几何头疼的地方一般在于代码量大和精度问题,代码量问题只要平时注意积累模 ...

  9. hdu 2393:Higher Math(计算几何,水题)

    Higher Math Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

随机推荐

  1. linux下手动安装git教程

    Git是一个开源的分布式版本控制系统,可以有效.高速的处理从很小到非常大的项目版本管理.而国外的GitHub和国内的Coding都是项目的托管平台.但是在使用git工具的时候,第一步要学会如何安装gi ...

  2. 配置nginx官网yum源

    由于yum源中没有我们想要的nginx,那么我们就需要创建一个“/etc/yum.repos.d/nginx.repo”的文件,其实就是新增一个yum源 二.添加nginx.repo 文件: [roo ...

  3. LintCode,hihoCoder,LeetCode有什么区别?

    https://www.zhihu.com/question/31218682 知乎用户 9 人赞同了该回答 LintCode 和LeetCode的题差不太多LintCode 有中文,不过没有用户讨论 ...

  4. 铁乐学python_day28_模块学习3

    大部份内容摘自授课老师的博客http://www.cnblogs.com/Eva-J/ OS模块复习一二 >>> import os >>> os.getcwd() ...

  5. matlab 函数句柄@的介绍_什么是函数句柄(转)

    http://blog.csdn.net/kevinhg/article/details/8861774 http://www.ilovematlab.cn/thread-30375-1-1.html ...

  6. sharepoint 2013 sp1 patch安装后的手工运行

    在安装SP1 后,有时Center admin 会显示 那么必须在以administrator运行sharepoint 2013 powershell. PSConfig.exe -cmd upgra ...

  7. 【2017-11-19】Linux基础知识:TP-Link WN823N无线网卡(RTL8192EU芯片)的X86-64及AARCH64驱动安装

    目的: 使类似于树莓派的AARCH-64架构的嵌入式设备能通过USB无线网卡连接上以太网: 该设备有LAN接口,但在前一次系统固件升级后,其内部的三个网络接口可以相互ping通,但任一接口无法ping ...

  8. python3: 字符串和文本(3)

    11. 删除字符串中不需要的字符  strip() 方法能用于删除开始或结尾的字符: lstrip() 和 rstrip() 分别从左和从右执行删除操作 >>> s = ' hell ...

  9. 如何让邮件营销平台成为EDM神器?

    任何一家做邮件营销的企业都希望自己的投入获得超乎想象的回报,出现打开率.点击率和伴随而来的成交量能够节节攀升的现象,这些数据我们当然可以通过监测各种平台的反馈而得到确切的报表.当然,作为邮件营销平台运 ...

  10. python decorator的本质

    推荐查看博客:python的修饰器 对于Python的这个@注解语法糖- Syntactic Sugar 来说,当你在用某个@decorator来修饰某个函数func时,如下所示: @decorato ...