zoukankan      html  css  js  c++  java
  • Luogu P1429 平面最近点对 【分治】By cellur925

    题目传送门

    题目大意:给定平面上n个点,找出其中的一对点的距离,使得在这n个点的所有点对中,该距离为所有点对中最小的。$n$<=100000。


    $Algorithm$

    最朴素的$n^2$枚举肯定是不行了,我们在这个数量级只能考虑$nlogn$做法。那么与这个数量级比较相关的也就是分治了。 把整个平面分为两个部分,分别求出两个部分点对间最小的距离,之后再处理跨区域的情况。

    分治法求解步骤: O(NlogN)  by hzwer
    1 将点集 S 分为两个⼦集 SL SR 分别求解
    2 δ 为⼦集中求得的最优值(min(δL; δR)),合并两个集合求
    解。
    图中以分界线为中⼼,任何⼀个 2δ · 2δ 的正⽅形内,只有常
    数个点,暴⼒ for 过去就好了。


    $Code$

     1 #include<cstdio>
     2 #include<algorithm>
     3 #include<cmath> 
     4 
     5 using namespace std;
     6 
     7 int n;
     8 int que[200090];
     9 struct node{
    10     double x,y;
    11 }p[200090];
    12 bool cmp(node a,node b)
    13 {
    14     return a.x<b.x;
    15 }
    16 
    17 bool cmp2(int a,int b)
    18 {
    19     return p[a].y<p[b].y;
    20 }
    21 
    22 double dis(int i,int j)
    23 {
    24     return sqrt((p[i].x-p[j].x)*(p[i].x-p[j].x)+(p[i].y-p[j].y)*(p[i].y-p[j].y));
    25 }
    26 
    27 double merge(int l,int r)
    28 {
    29     double dd=1e8;
    30     if(l==r) return dd;
    31     if(l+1==r) return dis(l,r);
    32     int mid=(l+r)>>1;
    33     double dl=merge(l,mid);
    34     double dr=merge(mid+1,r);
    35     dd=min(dl,dr);
    36     
    37     int pos=0;
    38     for(int i=l;i<=r;i++)
    39         if(fabs(p[mid].x-p[i].x)<dd) que[++pos]=i;
    40     sort(que+1,que+1+pos,cmp2);
    41     for(int i=1;i<=pos;i++)
    42         for(int j=i+1;j<=pos;j++)
    43         {
    44             if(dis(que[i],que[j])>dd) break;
    45             double ddd=dis(que[i],que[j]);
    46             dd=min(dd,ddd);
    47         }
    48     return dd;    
    49 }
    50 
    51 int main()
    52 {
    53     scanf("%d",&n);
    54     for(int i=1;i<=n;i++) scanf("%lf%lf",&p[i].x,&p[i].y);
    55     sort(p+1,p+1+n,cmp);
    56     printf("%.4lf",merge(1,n));
    57     return 0;
    58 }
    View Code

    几个注意事项

    • 边界的处理。
        if(l==r) return dd;
        if(l+1==r) return dis(l,r);
    • 利用正方形里有常数个点的性质时,要及时$break$。否则会超时。
    • 图中的$L$线大概是$mid$。开始找点的时候距离与他比较。
  • 相关阅读:
    人月神话阅读笔记之二
    第二阶段:团队开发Fooks第三天
    第二阶段:团队开发Fooks第二天
    第二阶段:团队开发Fooks第一天
    第十三周进度总结
    人月神话阅读笔记之一
    第一阶段意见评论
    拓扑排序
    欧拉回路
    矩阵优化
  • 原文地址:https://www.cnblogs.com/nopartyfoucaodong/p/9737687.html
Copyright © 2011-2022 走看看