题目

有 \(n\) 个圆$c_1,c_2, \cdots , c_n $,执行如下的操作:

找到剩下的半径最大的圆删除并删除所有和它有交的其他并没有被删除的圆;

求每个圆是被那个圆删除的;

$1 \le n \le 3 \times 10^5 $ ;

描述

  • kdt做法:

    记录每个圆围成的举行作为剪枝,直接模拟删除;

    记得旋转一下,然后eps开1e-3就好;

  • 搬运一下$n \ log^2n $做法(orz yww):

    考虑找到和\(c_i\) 相交的半径最大的被自己删除的圆 ;

    这样的圆一定满足互相不相交;

    由于半径比 \(c_i\) 大,所以如果相交一定会和 \(c_i\) 平行坐标轴的四条切线有交;

    所以用扫描线+cdq分治即可;

    //懒得写正解了这是暴力:
    #include<bits/stdc++.h>
    #define eps 1e-3
    #define ld double
    using namespace std;
    const int N=300010;
    const ld sq2=sqrt(2);
    int n,WD,pos[N],bl[N],ls[N],rs[N],tr[N],rt;///
    struct P{ld x,y;};///
    ld mn[N][2],mx[N][2];///
    struct C{P o;ld r,mn[2],mx[2];int id;}c[N];///
    int dcmp(ld x){return fabs(x)<eps?0:x<0?-1:1;}///
    bool cmp(C A,C B){return !dcmp(A.r-B.r)?A.id<B.id:A.r>B.r;}///
    bool cmpD(int A,int B){return !WD?c[A].o.x<c[B].o.x:c[A].o.y<c[B].o.y;}///
    ld len(P A,P B){return (A.x-B.x)*(A.x-B.x)+(A.y-B.y)*(A.y-B.y);}///
    bool judge(C A,C B){return dcmp(len(A.o,B.o)-(A.r+B.r)*(A.r+B.r))<=0;}///
    void rotate(P&A){A=(P){(A.x-A.y)/sq2,(A.x+A.y)/sq2};}///
    char gc(){
    static char*p1,*p2,s[1000000];
    if(p1==p2)p2=(p1=s)+fread(s,1,1000000,stdin);
    return(p1==p2)?EOF:*p1++;
    }///
    int rd(){
    int x=0,f=1;char C=gc();
    while(C<'0'||C>'9'){if(C=='-')f=-1;C=gc();}
    while(C>='0'&&C<='9'){x=x*10+C-'0';C=gc();}
    return x*f;
    }///
    void build(int&k,int l,int r,int d){
    WD=d;k=(l+r)>>1;
    nth_element(tr+l,tr+k,tr+r+1,cmpD);
    mn[k][0]=c[tr[k]].mn[0];
    mx[k][0]=c[tr[k]].mx[0];
    mn[k][1]=c[tr[k]].mn[1];
    mx[k][1]=c[tr[k]].mx[1];
    if(l<k){
    build(ls[k],l,k-1,d^1);
    for(int i=0;i<2;++i){
    mn[k][i]=min(mn[k][i],mn[ls[k]][i]);
    mx[k][i]=max(mx[k][i],mx[ls[k]][i]);
    }
    }
    if(k<r){
    build(rs[k],k+1,r,d^1);
    for(int i=0;i<2;++i){
    mn[k][i]=min(mn[k][i],mn[rs[k]][i]);
    mx[k][i]=max(mx[k][i],mx[rs[k]][i]);
    }
    }
    }///
    bool jud(int x,int y){return !x||mx[x][0]<c[y].mn[0]-eps||mn[x][0]>c[y].mx[0]+eps||mx[x][1]<c[y].mn[1]-eps||mn[x][1]>c[y].mx[1]+eps;}///
    void query(int k,int x){
    if(!bl[tr[k]]&&judge(c[tr[k]],c[x]))bl[tr[k]]=c[x].id;
    if(!jud(ls[k],x))query(ls[k],x);
    if(!jud(rs[k],x))query(rs[k],x);
    }///
    int main(){
    // freopen("B.in","r",stdin);
    // freopen("B.out","w",stdout);
    n=rd();
    for(int i=1;i<=n;++i){
    c[i].o.x=rd();c[i].o.y=rd();
    rotate(c[i].o);c[i].r=rd();
    c[i].mn[0]=c[i].o.x-c[i].r;
    c[i].mx[0]=c[i].o.x+c[i].r;
    c[i].mn[1]=c[i].o.y-c[i].r;
    c[i].mx[1]=c[i].o.y+c[i].r;
    c[i].id=tr[i]=i;
    }///
    sort(c+1,c+n+1,cmp);
    for(int i=1;i<=n;++i)pos[c[i].id]=i;
    build(rt,1,n,0);
    for(int i=1;i<=n;++i){
    if(!bl[i])query(rt,i);
    }
    for(int i=1;i<=n;++i)printf("%d ",bl[pos[i]]);
    return 0;//
    }//

【loj2586】【APIO2018】选圆圈的更多相关文章

  1. LOJ2586 APIO2018 选圆圈

    考前挣扎 KD树好题! 暴力模拟 通过kd树的结构把子树内的圈圈框起来 然后排个序根据圆心距 <= R1+R2来判断是否有交点 然后随便转个角度就可以保持优越的nlgn啦 卡精度差评 必须写ep ...

  2. 「APIO2018选圆圈」

    「APIO2018选圆圈」 题目描述 在平面上,有 \(n\) 个圆,记为 \(c_1, c_2, \ldots, c_n\) .我们尝试对这些圆运行这个算法: 找到这些圆中半径最大的.如果有多个半径 ...

  3. BZOJ5465 APIO2018选圆圈(KD-Tree+堆)

    考虑乱搞,用矩形框圆放KD-Tree上,如果当前删除的圆和矩形有交就递归下去删.为防止被卡,将坐标系旋转一定角度即可.注意eps稍微设大一点,最好开上long double. #include< ...

  4. [BZOJ5465][APIO2018]选圆圈(KD-Tree)

    题意:给你n个圆,每次选择半径最大的,将它和与它相交的圆全部删去,输出每个圆是在哪次被删的. KD树模板题.用一个矩形框住这个圆,就可以直接剪枝了.为了防止被卡可以将点旋转一个角度,为了保险还可以多转 ...

  5. 【LG4631】[APIO2018]Circle selection 选圆圈

    [LG4631][APIO2018]Circle selection 选圆圈 题面 洛谷 题解 用\(kdt\)乱搞剪枝. 维护每个圆在\(x.y\)轴的坐标范围 相当于维护一个矩形的坐标范围为\([ ...

  6. 【APIO2018】选圆圈(平面分块 | CDQ分治 | KDT)

    Description 给定平面上的 \(n\) 个圆,用三个参数 \((x, y, R)\) 表示圆心坐标和半径. 每次选取最大的一个尚未被删除的圆删除,并同时删除所有与其相切或相交的圆. 最后输出 ...

  7. 【LOJ2586】【APIO2018】选圆圈 CDQ分治 扫描线 平衡树

    题目描述 在平面上,有 \(n\) 个圆,记为 \(c_1,c_2,\ldots,c_n\) .我们尝试对这些圆运行这个算法: 找到这些圆中半径最大的.如果有多个半径最大的圆,选择编号最小的.记为 \ ...

  8. [loj2586]选圆圈

    下面先给出比较简单的KD树的做法-- 根据圆心建一棵KD树,然后模拟题目的过程,考虑搜索一个圆 剪枝:如果当前圆[与包含该子树内所有圆的最小矩形]都不相交就退出 然而这样的理论复杂度是$o(n^2)$ ...

  9. LOJ 2586 「APIO2018」选圆圈——KD树

    题目:https://loj.ac/problem/2586 只会 19 分的暴力. y 都相等,仍然按直径从大到小做.如果当前圆没有被删除,那么用线段树把 [ x-r , x+r ] 都打上它的标记 ...

随机推荐

  1. 绝对干货!初学者也能看懂的DPDK解析

    欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 本文由Willko发表于云+社区专栏 一.网络IO的处境和趋势 从我们用户的使用就可以感受到网速一直在提升,而网络技术的发展也从1GE/10 ...

  2. c++ 使用this指针进行串联的函数调用

    如代码所示,在每个成员函数函数体最后返回*this.即可实现串联调用. class Time { public: Time(, , ); Time &setHour(int); Time &a ...

  3. caffe 预训练 或者Fine-Tuning 操作

    1.使用预训练模型,需要修改训练的prototxt,将layer name改为与要使用模型的layer name相同即可. Borrowing Weights from a Pretrained Ne ...

  4. [东北师大软工]Week2-作业2:个人项目实战 初步测试结果

    作业地址 https://edu.cnblogs.com/campus/nenu/2016SE_NENU/homework/1656 测试须知 测试机为Windows环境,所有提交到Coding.ne ...

  5. 王者荣耀交流协会final发布文案美工展示博客

    logo: 我们的logo是蓝底白字,非常简洁大气的设计感,上面印有我们的软件名称,更好的直观的彰显了我们的主题.我们的软件就是要迎合使用者,给使用者更加方便快捷的工作体验,更好的衡量自己的时间分配. ...

  6. spring boot之配置跨域

    在启动类中配置 @Bean public WebMvcConfigurer corsConfigurer() { return new WebMvcConfigurer() { @Override p ...

  7. Daily Scrumming 2015.10.20(Day 1)

    一.今明两天任务表 Member Today’s Task Tomorrow’s Task 江昊 购买服务器,搭建服务器,配置服务器端用户与权限管理 配置ruby与rails环境 配置mysql与数据 ...

  8. 第二阶段Sprint冲刺会议3

     进展:讨论视频录制的具体功能,查看有关资料,开始着手编写有关代码.

  9. 利用python进行简单的图像处理:包括打开,显示以及保存图像

    利用python进行简单的图像处理:包括打开,显示以及保存图像 利用PIL处理 PIL(python image library) 是python用于图片处理的package.但目前这个package ...

  10. 软工网络15团队作业8——Beta阶段敏捷冲刺(day1)

    第 1 篇 Scrum 冲刺博客 1. 介绍小组新加入的成员,Ta担任的角色 --给出让ta担当此角色的理由 小组新加入的成员:3085叶金蕾 担任的角色:测试/用户体验/开发 理由:根据小组讨论以及 ...