题目大意

用最小矩形覆盖平面上所有的点

分析

有一结论:最小矩形中有一条边在凸包的边上,不然可以旋转一个角度让面积变小

简略证明

我们逆时针枚举一条边

用旋转卡壳维护此时最左,最右,最上的点

注意

注意凸包后点数不再是n

吐槽

凸包后点数是n,bzoj上就过了???

solution

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cctype>
#include <cmath>
#include <algorithm>
using namespace std;
typedef double db;
const db eps=1e-9;
const int M=50007; int n; struct pt{
db x,y;
pt(db _x=0.0,db _y=0.0){x=_x; y=_y;}
}p[M],s[M]; int tot; bool eq(db x,db y){return fabs(y-x)<=eps;}
bool le(db x,db y){return eq(x,y)||x<y;} pt operator -(pt x,pt y){return pt(x.x-y.x,x.y-y.y);}
pt operator +(pt x,pt y){return pt(x.x+y.x,x.y+y.y);}
bool operator <(pt x,pt y){return (x.y!=y.y)?(x.y<y.y):(x.x<y.x);}
bool operator ==(pt x,pt y){return eq(x.x,y.x)&&eq(x.y,y.y);};
pt operator *(pt x,db d){return pt(x.x*d,x.y*d);}
pt operator /(pt x,db d){return pt(x.x/d,x.y/d);} db dot(pt x,pt y){
return x.x*y.x+x.y*y.y;
} db cross(pt x,pt y){
return x.x*y.y-x.y*y.x;
} db length(pt x){
return sqrt(dot(x,x));
} db area(pt x,pt y,pt z){
return cross(y-x,z-x);
} db shadow(pt x,pt y,pt to){
return dot(y-x,to-x)/length(to-x);
} pt lf_90(pt x){
return pt(-x.y,x.x);
} bool cmp(pt x,pt y){
db tp=area(p[1],x,y);
if(eq(tp,0)) return length(x-p[1])<length(y-p[1]);
return tp>0;
} void convex(){
int i,ii=1;
for(i=2;i<=n;i++) if(p[i]<p[ii]) ii=i;
swap(p[1],p[ii]);
sort(p+2,p+n+1,cmp); s[tot=1]=p[1];
for(i=2;i<=n;i++){
while(tot>1&&le(area(s[tot-1],s[tot],p[i]),0)) tot--;
s[++tot]=p[i];
}
} int main(){
int i,p1,p2,p3;
db tp1,tp2,tp3,tp4,ans;
pt a[5],tp; scanf("%d",&n); for(i=1;i<=n;i++) scanf("%lf%lf",&p[i].x,&p[i].y); convex(); s[0]=s[tot];//要算每一条边,加上tot-0的 ans=1e32;
p1=1,p2=1,p3=1; for(i=0;i<tot;i++){
if(s[i]==s[i+1]) continue; while(le(area(s[i],s[i+1],s[p3]),area(s[i],s[i+1],s[p3%tot+1]))) p3=p3%tot+1;
if(i==0) p1=p3;//第一次找卡壳特例
while(le(shadow(s[i],s[p1%tot+1],s[i+1]),shadow(s[i],s[p1],s[i+1]))) p1=p1%tot+1;
while(le(shadow(s[i+1],s[p2%tot+1],s[i]),shadow(s[i+1],s[p2],s[i]))) p2=p2%tot+1; tp1=length(s[i+1]-s[i]);
tp2=area(s[i],s[i+1],s[p3])/tp1;
tp3=fabs(shadow(s[i],s[p1],s[i+1]));
tp4=fabs(shadow(s[i+1],s[p2],s[i])); if(le((tp1+tp3+tp4)*tp2,ans)){
ans=(tp1+tp3+tp4)*tp2;
tp=s[i+1]-s[i];
a[1]=s[i]-tp*(tp3/tp1);
a[2]=s[i+1]+tp*(tp4/tp1);
tp=lf_90(tp);
a[3]=a[2]+tp*(tp2/tp1);
a[4]=a[1]+tp*(tp2/tp1);
}
} printf("%.5lf\n",ans+eps); int ii=1;
for(i=2;i<=4;i++) if(a[i]<a[ii]) ii=i;
printf("%.5lf %.5lf\n",a[ii].x+eps,a[ii].y+eps);
for(i=ii%4+1;i!=ii;i=i%4+1) printf("%.5lf %.5lf\n",a[i].x+eps,a[i].y+eps); return 0;
}

最新文章

  1. PHP二维数组排序(list_order)
  2. Safari 快捷键
  3. ajax同步、异步执行简单理解与证明
  4. Windows KB2984972安装后堵住了一个windows 7 桌面可以多个用户远程访问桌面的漏洞。
  5. CentOS6.3下安装VSFTP服务
  6. c# 6.0新特性(二)
  7. java集合框架之java HashMap代码解析
  8. poj2186 强连通
  9. JVM学习总结五(番外)——VisualVM
  10. 俄罗斯方块游戏 --- java
  11. 站点建设10个最好的响应的HTML5滑块插件
  12. Magento How To Display Product Custom Option On list.phtml
  13. Linux中samba服务器的搭建
  14. 微信小程序简单入门理解
  15. 读《图解HTTP》有感-(HTTP首部)
  16. Lily_music 网页音乐播放器 -可搜索(附歌词联动播放效果解说)
  17. 关于HTMl CSS
  18. Day6 Python常用的模块
  19. POJ 2235 Frogger / UVA 534 Frogger /ZOJ 1942 Frogger(图论,最短路径)
  20. 20165318 2017-2018-2 《Java程序设计》第三周学习总结

热门文章

  1. IEDA的安装与破解
  2. JavaScript -- 内置对象数组
  3. css实现页面文字不换行、自动换行、强制换行
  4. Yum简单使用小结
  5. 学习笔记(三): Generalization/Overfitting/Validation
  6. 【最大流】bzoj1711: [Usaco2007 Open]Dining吃饭
  7. i2c drivers
  8. LeetCode(134) Gas Station
  9. holtek编程注意事项
  10. Apache ant 配置