1069: [SCOI2007]最大土地面积

准备开始BZOJ补完计划了。

话说终于可以切计算几何了,好开森。

虽然这题确实很简单。。。。。

第一眼就看出来 凸包+旋转卡壳没跑了。

先求凸包。

然后在旋转卡壳的时候,每次求出一对对踵点,在连线的左侧和右侧各找一个最远点,四个点构成的可能是最大面积,求一下更新。

具体证明什么的??????

不会(。・_・。)ノ

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
const double eps=1e-8;
int dcmp(double x){
	if(fabs(x)<eps)return 0;
	return x<0?-1:1;
}
struct point{
	double x,y;
};
bool cmp(point a,point b){
	if(!dcmp(a.x-b.x))return a.y<b.y;
	return a.x<b.x;
}
typedef point vector;
vector operator-(point a,point b){
	return (vector){a.x-b.x,a.y-b.y};
}
double cross(vector a,vector b){
	return a.x*b.y-a.y*b.x;
}
double cross(point a,point b,point c){
	return cross(b-a,c-a);
}
double sqr(double x){return x*x;}
double dist(point a,point b){
	return sqrt(sqr(a.x-b.x)+sqr(a.y-b.y));
}
struct ConvexHull{
	point ch[2005];
	int m;
	void andrew(point *p,int n){
		sort(p+1,p+1+n,cmp);
		m=0;
		for(int i=1;i<=n;i++){
			while(m>1&&cross(ch[m-2],ch[m-1],p[i])<=0)m--;
			ch[m++]=p[i];
		}
		int k=m;
		for(int i=n-1;i>=1;i--){
			while(m>k&&cross(ch[m-2],ch[m-1],p[i])<=0)m--;
			ch[m++]=p[i];
		}
		if(n)m--;
	}
	double up(point a,point b){
		double ans=0;
		for(int i=0;i<m;i++)
		ans=max(ans,cross(a,b,ch[i])*0.5);
		return ans;
	}
	double down(point a,point b){
		double ans=0;
		for(int i=0;i<m;i++)
		ans=min(ans,cross(a,b,ch[i])*0.5);
		return -ans;
	}
	double rotate(){
		int j=1;
		double ans=0;
		for(int i=0;i<m;i++){
			while(cross(ch[i],ch[i+1],ch[j])<cross(ch[i],ch[i+1],ch[j+1]))
			j=(j+1)%m;
			ans=max(ans,up(ch[i],ch[j])+down(ch[i],ch[j]));
		}
		return ans;
	}
}sol;
point p[2005];
int main(){
	int n;scanf("%d",&n);
	for(int i=1;i<=n;i++)scanf("%lf%lf",&p[i].x,&p[i].y);
	sol.andrew(p,n);
	printf("%.3lf",sol.rotate());
	return 0;
}
	 


你可能感兴趣的:(1069: [SCOI2007]最大土地面积)