zoukankan      html  css  js  c++  java
  • 【UVALive 4642】Malfatti Circles(圆,二分)

    给定三角形,求三个两两相切且与三角形的一条边相切的圆的半径。

    二分一个半径,可以得出另外两个半径,需要推一推公式(太久了,我忘记了)

    #include<cstdio>
    #include<cmath>
    #define eps (1e-8)
    #define sqr(a) (a)*(a)
    #define min(a,b) (a)>(b)?(b):(a)
    #define dd double
    struct point{
        dd x,y,v,a;//点x,y,v为角度,a为边长
    }q[10];
    dd r0,r1,r2,r3;
    int read(){
        int f=0;
        for(int i=0;i<3;i++){
            scanf("%lf%lf",&q[i].x,&q[i].y);
            if(q[i].x||q[i].y) f=1;
        }
        return f;
    }
    dd make(dd r,dd a,dd h,dd n){
        dd t=r-r/tan(h)/tan(n)+a/tan(n);
        if(t<=eps)return -1;
        return sqr((sqrt(t)-sqrt(r))*tan(n));
    }
    void solve(){
        dd l=0,r=min(q[0].a*tan(q[0].v),q[2].a*tan(q[2].v));
        while(r-l>eps){
            r0=(l+r)/2;
            r1=make(r0,q[0].a,q[0].v,q[1].v);
            r2=make(r0,q[2].a,q[0].v,q[2].v);
            if(r1<=eps||r2<=eps||r2/tan(q[2].v)+r1/tan(q[1].v)+2*sqrt(r2*r1)-q[1].a<eps)
                r=r0;
            else
                l=r0;
        }
    }
    int main(){
        while(read())
        {
            for(int i=0;i<3;i++)
                q[i].a=sqrt(sqr(q[(i+1)%3].x-q[i].x)+sqr(q[(i+1)%3].y-q[i].y));//计算边长
            for(int i=0;i<3;i++)
                q[i].v=acos((sqr(q[i].a)+sqr(q[(i+2)%3].a)-sqr(q[(i+1)%3].a))/2/q[i].a/q[(i+2)%3].a)/2;
            solve();
            printf("%lf %lf %lf
    ",r0,r1,r2);
        }
    }

      

  • 相关阅读:
    UOJ#424. 【集训队作业2018】count
    框架的 总结(nop)------添加功能
    c# 调用分页(控制器端的)与时间的格式
    c# 通过关键字查询
    c#导入导出 插入数据到用户表(nop框)
    Nop权限的使用
    联合结果集的原则
    简单的结果集联合
    UNION ALL
    联合结果集
  • 原文地址:https://www.cnblogs.com/flipped/p/5723253.html
Copyright © 2011-2022 走看看