ccz181078

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: :: 管理 ::

Description

给定一个n个点的严格凸多边形(各个内角<180°),现在要切出两个非退化三角形(三点不共线),要求两个三角形顶点必须是凸多边形的顶点,且三角形不可相交(但是点或边可以重合)。求两个三角形面积之差的最大值。

Input

第一行,一个整数N。
第二到N+1行,每行两个整数xi,yi,表示多边形的一个点,保证顶点按顺时针或逆时针顺序给出。

Output

输出答案,精确到小数点后1位。
最小三角形一定是相邻三点构成,因此只需枚举每个最小三角形,查询剩余部分中的最大三角形
dp,f[a][b]表示三角形一条边从a到b的,另一个顶点c在a到b之间的最大面积,固定a,令b递增则决策点c可以线性求出
辅助g[a][b]=max(g[a][b-1],g[a+1][b],f[a][b])可支持查询
#include<bits/stdc++.h>
typedef long long i64;
struct pos{int x,y;}ps[10007];
pos operator+(pos a,pos b){return (pos){a.x+b.x,a.y+b.y};}
pos operator-(pos a,pos b){return (pos){a.x-b.x,a.y-b.y};}
i64 operator*(pos a,pos b){return i64(a.x)*b.y-i64(a.y)*b.x;}
int n,ws[5007];
i64 ans=0,s0[5007],ms[5007];
void maxs(i64&a,i64 b){if(a<b)a=b;}
int main(){
    scanf("%d",&n);
    for(int i=1;i<=n;++i)scanf("%d%d",&ps[i].x,&ps[i].y);
    if((ps[2]-ps[1])*(ps[3]-ps[1])<0)std::reverse(ps+1,ps+n+1);
    ps[n*2+1]=ps[1];
    ps[0]=ps[n];
    for(int i=1;i<=n;++i)ps[n+i]=ps[i],ws[i]=i;
    for(int d=3;d<n;++d){
        for(int l=1,r=d;l<=n;++l,++r){
            pos a=ps[r]-ps[l];
            s0[l]=(ps[ws[l]]-ps[l])*a;
            while(ws[l]+1<r){
                i64 s1=(ps[ws[l]+1]-ps[l])*a;
                if(s1<s0[l])break;
                ++ws[l];
                s0[l]=s1;
            }
            maxs(ms[l],s0[l]);
        }
        ms[n+1]=ms[1];
        for(int l=1;l<=n;++l)maxs(ms[l],ms[l+1]);
    }
    for(int l=1,r=n-1;l<=n;++l,++r)maxs(ans,ms[l]-(ps[r]-ps[l])*(ps[r+1]-ps[l]));
    printf("%lld.%lld\n",ans/2,ans%2*5);
    return 0;
}

 

posted on 2017-07-06 23:58  nul  阅读(271)  评论(0编辑  收藏  举报