BZOJ2458 Beijing2011最小三角形(分治)

  类似于平面最近点对,考虑分治,即分别计算分割线两侧的最小三角形再考虑跨过线的三角形。

  复杂度证明也是类似的,对于某一个点,在另一侧可能与其构成最小三角形的点在一个d*d/2的矩形内(两边之和大于第三边),并且这些点所组成的三角形周长均不小于d。然而并不清楚这里至多会有多少个点,vfk曾说上界是16,我当然不会证明这个上界也构造不出来有这么多点的方案。找这些点的时候归并就可以做到线性。那么复杂度是O(nlogn)乘上枚举这些点的常数2*16*15/2,看起来根本跑不动不过这个上界肯定是特别松的所以一点也不虚。

  算距离的时候会爆int。以及luogu数据疑似有锅。

#include<iostream> 
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
int read()
{
    int x=0,f=1;char c=getchar();
    while (c<'0'||c>'9') {if (c=='-') f=-1;c=getchar();}
    while (c>='0'&&c<='9') x=(x<<1)+(x<<3)+(c^48),c=getchar();
    return x*f;
}
#define N 200010
#define inf 1000000000
int n;
double ans=inf;
struct data
{
    int x,y;
    bool operator <(const data&a) const
    {
        return x<a.x||x==a.x&&y<a.y;
    }
    double operator -(const data&a) const
    {
        return sqrt(1ll*(x-a.x)*(x-a.x)+1ll*(y-a.y)*(y-a.y));
    }
}a[N],b[N],c[N];
void getans(data *b,data *c,int n,int m)
{
    int s=1,t=0;
    for (int i=1;i<=n;i++)
    {
        while (s<=m&&c[s].y+ans/2<b[i].y) s++;
        while (t<m&&c[t+1].y-ans/2<b[i].y) t++;
        for (int j=s;j<t;j++)
        {
            double tot=b[i]-c[j];
            for (int k=j+1;k<=t;k++)
            ans=min(ans,tot+(c[k]-b[i])+(c[k]-c[j]));
        }
    }
}
void solve(int l,int r)
{
    if (l==r) return;
    int mid=l+r>>1;
    solve(l,mid);
    solve(mid+1,r);
    int n=0,m=0,MID=-inf;
    for (int i=l;i<=mid;i++) MID=max(MID,a[i].x);
    for (int i=l;i<=mid;i++)
    if (2*(MID-a[i].x)<ans) b[++n]=a[i];
    for (int i=mid+1;i<=r;i++)
    if (2*(a[i].x-MID)<ans) c[++m]=a[i];
    getans(b,c,n,m);
    getans(c,b,m,n);
    int i=l,j=mid+1;
    for (int k=l;k<=r;k++)
    if (j>r||i<=mid&&a[i].y<a[j].y) b[k]=a[i++];
    else b[k]=a[j++];
    for (int k=l;k<=r;k++) a[k]=b[k];
}
int main()
{
#ifndef ONLINE_JUDGE
    freopen("bzoj2458.in","r",stdin);
    freopen("bzoj2458.out","w",stdout);
    const char LL[]="%I64d\n";
#else
    const char LL[]="%lld\n";
#endif
    n=read();
    for (int i=1;i<=n;i++) a[i].x=read(),a[i].y=read();
    sort(a+1,a+n+1);
    solve(1,n);
    printf("%.6lf",ans);
    return 0;
}

 

posted @ 2018-09-02 18:19  Gloid  阅读(132)  评论(0编辑  收藏  举报