zoukankan      html  css  js  c++  java
  • HDU 5531

    题目大意:

    给定一个n边形的顶点

    以每个顶点为圆心画圆(半径可为0)

    每个顶点的圆要和它相邻顶点的圆相切(不相邻的可相交)

    求所有圆的最小面积总和并给出所有圆的半径

    设半径为r1 r2 ... rn,顶点距离为L1 L2 ... Ln

    当顶点数为奇数时 由

    r1+r2=L1

    r2+r3=L2

    ......

    rn+r1=Ln

    可得 r1+r1=L1-L2+L3-......+Ln

    只要中间所有半径 r >= 0 那么r1就有唯一解

    当顶点数为偶数时 由

    r1+r2=L1

    r2+r3=L2

    ......

    rn+r1=Ln

    可得 0=L1-L2+L3-......-Ln

    即 L1+L3+...+Ln-1=L2+L4+...+Ln 时 才有解

    由 上式可得

    r2=L1-r1

    r3=L2-L1+r1

    ......

    r1=Ln-......+r1

    也就是每个 r 都可以转换为 

    第偶数个顶点时 r=x+r1 或

    第奇数个顶点时 r=x-r1 的形式

    已知 r>=0 得 

    第偶数个顶点时 x+r1>=0 即 r1>=-x 

    第奇数个顶点时 x-r1>=0 即 r1<=x

    那么由所有顶点的约束 就能得到r1的范围 [L,R]

    圆的面积总和 S=PI*(r1^2+r2^2+r3^2+...+rn^2)

    而每个r^2都可以转换为 r^2=(r1+x)^2  或 r^2=(r1-x)^2

    如: r2^2=(r1-L1)^2, r3^2=(r1+L2-L1)^2

    即 递推时 x=Li-x

    第偶数个顶点时 ri^2=(r1-x)^2

    第奇数个顶点时 ri^2=(r1+x)^2

    那么 S=PI*(a*r1^2+b*r1+c)

    如: r2^2=(r1-L1)^2, r3^2=(r1+L2-L1)^2

        ​    ​则 S1=PI*(1*r1^2+0*r1+0) (a=1,b=0,c=0,x=0)

        ​    ​S2=PI*(S1+r2^2)=PI*(S1+(r1-x)^2)

        ​    ​    ​=PI*(S1+r1^2-2*x*r1+x*x) (a=a+1,b=b+(-1)*2*x,c=c+x*x,x=L1-x)

        ​    ​S3=PI*(S2+r3^2)=PI*(S2+(r1+x)^2)

        ​    ​    ​=PI*(S1+r1^2+2*x*r1+x*x) (a=a+1,b=b+(1)*2*x,c=c+x*x,x=L2-x)

    第偶数个顶点时 a+=1,b+=(-1)*2*x,c+=x*x,x=Li-x

    第奇数个顶点时 a+=1,b+=1*2*x,c+=x*x,x=Li-x

    又 a*r1^2+b*r1+c=0 时存在极值点 1-b/(2*a)

    再与r1的范围比较一下 就能得到r1 从而得到S的最小值

    或者直接利用[L,R] 用三分法求解

    #include <bits/stdc++.h>
    #define ll long long
    #define INF 0x3f3f3f3f
    using namespace std;
    
    const double eps = 1e-10;
    const double PI = acos(-1.0);
    double add(double a,double b) {
        if(abs(a+b)<eps*(abs(a)+abs(b))) return 0;
        return a+b;
    }
    struct P {
        double x,y;
        P(){};
        P(double _x,double _y):x(_x),y(_y){}
        P operator - (P p) {
            return P(add(x,-p.x),add(y,-p.y));
        }
        double dot(P p) {
            return add(x*p.x,y*p.y);
        }
    }p[10005];
    int n;
    double ans;
    double r[10005], len[10005];
    double lenV(P p) {
        return sqrt(p.dot(p));
    }
    double getArea(double r0) {
        double ans=0; r[0]=r0;
        for(int i=0;i<n;i++) {
            if(r[i]<-eps) return 0;
            ans+=r[i]*r[i];
            r[i+1]=len[i]-r[i];
            //printf("len[i]%lf r[i]%lf
    ",len[i],r[i]);
        }
        return ans*PI;
    }
    
    bool solve() {
        ll sum=0;
        for(int i=0;i<n;i++) {
            len[i]=lenV(p[(i+1)%n]-p[i]);
            sum+= i%2 ? -len[i]:len[i];
        }
        if(n&1) {
            double r0=sum/2.0;
            if(r0<-eps) return 0;
            ans=getArea(r0);
            //printf("ans = %lf
    ",ans);
        } else {
            if(sum) return 0;
            ll a,b,c;
            a=b=c=0;
            ll t=0,L=0,R=INF,k=1;
            for(int i=0;i<n;i++) {
                //printf("%lld %lld %lld
    ",a,b,c);
                a++;
                b+=2.0*k*t;
                c+=t*t;
                if(k==1) L=max(L,-t);
                else R=min(R,t);
                k*=-1;
                t=len[i]-t;
            }
            if(L>R) return 0;
            double mid=-b/2.0/a;
            mid=max(mid,(double)L);
            mid=min(mid,(double)R);
            ans=getArea(mid);
        }
        if(ans<eps) return 0;
        return 1;
    }
    
    int main()
    {
        int t; scanf("%d",&t);
        while(t--) {
            scanf("%d",&n);
            for(int i=0;i<n;i++)
                scanf("%lf%lf",&p[i].x,&p[i].y);
            if(solve()) {
                printf("%.2f
    ",ans);
                for(int i=0;i<n;i++)
                    printf("%.2f
    ",r[i]);
            } else printf("IMPOSSIBLE
    ");
        }
    
        return 0;
    }
    View Code
  • 相关阅读:
    再战MFC中的消息机制
    指针到底是由谁决定的
    静态初始化的一些东西
    人生少走弯路
    win32 程序分析
    毕业之前要做好这些准备
    今天写二叉树秀逗了~~~
    How to declare global variables in Android? --- Application Subclasses
    android Application类的详细介绍
    Android setTag方法的key问题
  • 原文地址:https://www.cnblogs.com/zquzjx/p/9763922.html
Copyright © 2011-2022 走看看