zoukankan      html  css  js  c++  java
  • 最小矩形覆盖

    #include<bits\stdc++.h>
    using namespace std;
    #define int long long
    void in(int &x){
        int    y=1;char c=getchar();x=0;
        while(c<'0'||c>'9'){if(c=='-')y=-1;c=getchar();}
        while(c<='9'&&c>='0'){ x=(x<<1)+(x<<3)+c-'0';c=getchar();}
        x*=y;
    }
    const int _ = 1e6+10;
    const double eps = 1e-9;
    const double pi = cos(-1);
    struct PT{
        double x,y;
        PT(double x=0,double y=0):x(x),y(y){}
        void input(){scanf("%lf%lf",&x,&y);}
        // 关键字排序
        bool operator<(const PT&p)const{
            if(fabs(x-p.x))return x<p.x;
                return y<p.y;
        }
        void output(){printf("%.10f %.10f\n",x,y);}
    }p[_],q[_];
    vector<PT>ret;
    int n;
    
    // 三点求叉积
    double vect(PT p,PT p1,PT p2){
        return (p1.x-p.x)*(p2.y-p.y)-(p1.y-p.y)*(p2.y-p.y);
    }
    int  convex_hull(PT* p,int n,PT *q){
        int i,k,m;
        sort(p,p+n);
        m = 0;
        for(i=0;i<n;q[m++]=p[i++])while(m>1&&vect(q[m-2],q[m-1],p[i])<eps)m--;
        k=m;
        for(i=n-2;i>=0;q[m++]=p[i--])while(m>k&&vect(q[m-2],q[m-1],p[i])<eps)m--;
        return --m;
    }
    // ?
    PT get(PT p,double x){
        return PT(p.x*cos(x)-p.y*sin(x),p.x*sin(x)+p.y*cos(x));
    }
    // ?
    bool is_ext(int id,PT pp){
        if(
                vect(p[id],PT(p[id].x+pp.x,p[id].y+pp.y)
                     ,p[id+1])<-eps
                     )return 0;
        if(
                vect(p[id],PT(p[id].x+pp.x,p[id].y+pp.y),
                     p[(id-1+n)%n])<-eps
                     )return 0;
        return 1;
    }
    PT inter(PT p1,PT p2,PT p3,PT p4){
        p2.x+=p1.x;
        p2.y+=p1.y;
        p4.x+=p3.x;
        p4.y+=p3.y;
        double s=vect(p1,p2,p3),s1=vect(p1,p2,p4);
        double t=s/(s-s1);
        return PT(p3.x+(p4.x-p3.x)*t,p3.y+(p4.y-p3.y)*t);
    }
    double ans;
    void solve(){
        int f[4];
        f[1]=f[2]=f[3]=0;
        for(int i=0;i<n;i++){
            f[0]=i;
    
            PT v[4];
            // 0-1的边
            v[0]=PT(p[i+1].x-p[i].x,p[i+1].y-p[i].y);
            //
            for(int j=1;j<4;j++)
                // 每个额外的点
                for(v[j]=get(v[0],pi/2*j);// v[j] 为 v[0] 选择 90°*j
                !is_ext(f[j],v[j]);
                f[j]=(f[j]+1)%n);
            vector<PT>tmp;
    
            for(int j=0;j<4;j++)tmp.push_back(inter(p[f[j]],v[j],p[f[(j+1)%4]],v[(j+1)%4]));
            double tmps=0;
    
            for(int j=0;j<4;j++)tmps+=vect(tmp[0],tmp[j],tmp[(j+1)%4]);
            tmps=fabs(tmps);
            if(ans>tmps)ans=tmps,ret=tmp;
        }
    }
    signed main(){
        while(~scanf("%d",&n)){
            if(!n)return 0;
            for(int i=0;i<n;i++)p[i].input();
            n=convex_hull(p,n,q);
            if(n<3)ans=0;else{
                    for(int i=0;i<n;i++)p[i]=q[i];
                    p[n]=p[0];
                    ans=1e100;
                    solve();
            }
            printf("%.4f\n",ans/2.0);
            //for(int i=0;i<4;i++)ret[i].output();
        }
    
        return 0;
    }
    
    
  • 相关阅读:
    CentOS6下OpenLDAP+PhpLdapAdmin基本安装及主从/主主高可用模式部署记录
    Curl运维命令
    kafka 基础知识梳理及集群环境部署记录
    RocketMQ 简单梳理 及 集群部署笔记
    Redis哨兵模式(sentinel)学习总结及部署记录(主从复制、读写分离、主从切换)
    Redis+Keepalived高可用环境部署记录
    Centos下Nodejs+npm环境-部署记录
    基于Ping和Telnet/NC的监控脚本案例分析
    linux dig 命令使用方法
    dnsjava usage
  • 原文地址:https://www.cnblogs.com/yesuweiYYYY/p/15546702.html
Copyright © 2011-2022 走看看