BZOJ1185[HNOI2007] 最小矩形覆盖

题面

给定一些点的坐标,要求求能够覆盖所有点的最小面积的矩形,输出所求矩形的面积和四个顶点的坐标

分析

首先可以先求凸包,因为覆盖了凸包上的顶点,凸包内的顶点也一定能被覆盖

结论:这个矩形的一条边一定与凸包的一条边重合。

然后对于凸包的每一条边\(\vec{s_is_{i+1}}\),我们通过旋转卡壳找到最左侧的点l,最右侧的点r,最高点p,过p做\(\vec{s_is_{i+1}}\)的平行线,过l,r做\(\vec{s_is_{i+1}}\)的垂线,就得到了我们要求的矩形

求最大高度的点用叉积最大,最右侧点用点积最大,最左侧点用点积最小(点积为负)

求坐标,可以先根据前面的点积和叉积求出投影长度,再求出边对应的单位向量,乘上长度再相加就得到了点的坐标。如l的坐标就是$s_i+ \frac{\vec{s_is_{i+1}} \cdot \vec{s_il}{}}{|\vec{s_is_{i+1}}|} \frac{ \vec{s_is_{i+1}} }{|\vec{s_is_{i+1}}|} $,另两个端点只需要向量旋转90度即可

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#define maxn 100000
#define eps 1e-10
#define INF 1e20
using namespace std;
int n;
struct Vector{
double x;
double y;
Vector(){ }
Vector(double _x,double _y){
x=_x;
y=_y;
}
friend Vector operator + (Vector p,Vector q){
return Vector(p.x+q.x,p.y+q.y);
}
friend Vector operator - (Vector p,Vector q){
return Vector(p.x-q.x,p.y-q.y);
}
friend Vector operator * (Vector a,double d){
return Vector(a.x*d,a.y*d);
}
friend Vector operator / (Vector a,double d){
return Vector(a.x/d,a.y/d);
}
};
typedef Vector point;
inline double dot(Vector p,Vector q){
return p.x*q.x+p.y*q.y;
}
inline double dist(point p,point q){
return sqrt(dot(p-q,p-q));
}
inline double cross(Vector p,Vector q){
return p.x*q.y-p.y*q.x;
}
inline double length(Vector x){
return sqrt(dot(x,x));
}
point a[maxn+5]; point O;
int cmp(point P,point Q){
double ang=cross(P-O,Q-O);
if(fabs(ang)<eps) return dist(O,P)<dist(O,Q);
else return ang>eps;
} int top=0;
point s[maxn+5];
void Graham(){
for(int i=1;i<=n;i++){
if(a[i].x<a[1].x||(a[i].x==a[1].x&&a[i].y<=a[1].y)) swap(a[i],a[1]);
}
O=a[1];
sort(a+2,a+1+n,cmp);
for(int i=1;i<=n;i++){
while(top>1&&cross(s[top]-s[top-1],a[i]-s[top-1])<=eps) top--;//在逆时针方向,叉积<0
s[++top]=a[i];
}
//#ifdef DEBUG
// for(int i=1;i<=top;i++){
// printf("(%.5f,%.5f)\n",s[i].x,s[i].y);
// }
//#endif
} inline int nex(int x){
return x%top+1;
}
inline Vector rotate90(Vector p){//把向量逆时针旋转90度
//(xcos(a)-ysin(a),xsin(a)+ycos(a)) a=pi/2
return Vector(-p.y,p.x);
} point res[10];
double ans=INF;
void Spin(){
s[top+1]=s[1];
int l,r,p;
l=r=p=2;
for(int i=1;i<=top;i++){
double D=length(s[i+1]-s[i]);
while(cross(s[i+1]-s[i],s[p+1]-s[i])-cross(s[i+1]-s[i],s[p]-s[i])>-eps) p=nex(p);
while(dot(s[r+1]-s[i],s[i+1]-s[i])-dot(s[r]-s[i],s[i+1]-s[i])>-eps) r=nex(r);
if(i==1) l=r;
while(dot(s[l+1]-s[i],s[i+1]-s[i])-dot(s[l]-s[i],s[i+1]-s[i])<eps) l=nex(l);
//在左侧的时候点积为负数,最小
double lenl=dot(s[l]-s[i],s[i+1]-s[i])/D;
double lenr=dot(s[r]-s[i],s[i+1]-s[i])/D;
double height=cross(s[i+1]-s[i],s[p]-s[i])/D;
double area=(fabs(lenr)+fabs(lenl))*fabs(height);//lenl,lenr是有方向的
if(area<ans){
// printf("(%.3f %.3f) ",(double)s[i].x,(double)s[i].y);
// printf("(%.3f %.3f)\n",(double)s[i+1].x,(double)s[i+1].y);
// printf("S=%.4f l=%d r=%d p=%d\n",area,l,r,p);
ans=area;
res[1]=s[i]+(s[i+1]-s[i])/D*lenl;
res[2]=s[i]+(s[i+1]-s[i])/D*lenr;
res[3]=res[1]+rotate90((s[i+1]-s[i])/D)*height;
res[4]=res[2]+rotate90((s[i+1]-s[i])/D)*height;
}
}
} int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%lf %lf",&a[i].x,&a[i].y);
}
Graham();
Spin();
for(int i=1;i<=4;i++){
if(res[i].y<res[1].y||(res[i].y==res[1].y&&res[i].x<res[1].x)) swap(res[1],res[i]);
}
O=res[1];
sort(res+2,res+1+4,cmp);
for(int i=1;i<=4;i++){
if(fabs(res[i].x)<eps) res[i].x=0;
if(fabs(res[i].y)<eps) res[i].y=0;
}
printf("%.5f\n",ans);
for(int i=1;i<=4;i++){
printf("%.5f %.5f\n",res[i].x,res[i].y);
}
}

BZOJ1185[HNOI2007] 最小矩形覆盖(旋转卡壳)的更多相关文章

  1. bzoj1185 [HNOI2007]最小矩形覆盖 旋转卡壳求凸包

    [HNOI2007]最小矩形覆盖 Time Limit: 10 Sec  Memory Limit: 162 MBSec  Special JudgeSubmit: 2081  Solved: 920 ...

  2. BZOJ 1185: [HNOI2007]最小矩形覆盖 [旋转卡壳]

    1185: [HNOI2007]最小矩形覆盖 Time Limit: 10 Sec  Memory Limit: 162 MBSec  Special JudgeSubmit: 1435  Solve ...

  3. 洛谷 P3187 BZOJ 1185 [HNOI2007]最小矩形覆盖 (旋转卡壳)

    题目链接: 洛谷 P3187 [HNOI2007]最小矩形覆盖 BZOJ 1185: [HNOI2007]最小矩形覆盖 Description 给定一些点的坐标,要求求能够覆盖所有点的最小面积的矩形, ...

  4. 【bzoj1185】[HNOI2007]最小矩形覆盖 (旋转卡壳)

    给你一些点,让你用最小的矩形覆盖这些点 首先有一个结论,矩形的一条边一定在凸包上!!! 枚举凸包上的边 用旋转卡壳在凸包上找矩形另外三点... 注意精度问题 #include<cstdio> ...

  5. bzoj 1185 [HNOI2007]最小矩形覆盖——旋转卡壳

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1185 矩形一定贴着凸包的一条边.不过只是感觉这样. 枚举一条边,对面的点就是正常的旋转卡壳. ...

  6. BZOJ 1185: [HNOI2007]最小矩形覆盖-旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标-备忘板子

    来源:旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标 BZOJ又崩了,直接贴一下人家的代码. 代码: #include"stdio.h" #include"str ...

  7. BZOJ1185 [HNOI2007]最小矩形覆盖 【旋转卡壳】

    题目链接 BZOJ1185 题解 最小矩形一定有一条边在凸包上,枚举这条边,然后旋转卡壳维护另外三个端点即可 计算几何细节极多 维护另外三个端点尽量不在这条边上,意味着左端点尽量靠后,右端点尽量靠前, ...

  8. 2018.10.18 bzoj1185: [HNOI2007]最小矩形覆盖(旋转卡壳)

    传送门 不难看出最后的矩形一定有一条边与凸包某条边重合. 因此先求出凸包,然后旋转卡壳求出当前最小矩形面积更新答案. 代码: #include<bits/stdc++.h> #define ...

  9. BZOJ1185 HNOI2007 最小矩形覆盖 凸包、旋转卡壳

    传送门 首先,肯定只有凸包上的点会限制这个矩形,所以建立凸包. 然后可以知道,矩形上一定有一条边与凸包上的边重合,否则可以转一下使得它重合,答案会更小. 于是沿着凸包枚举这一条边,通过旋转卡壳找到离这 ...

随机推荐

  1. 细数不懂Spring底层原理带来的伤与痛

    原文链接:https://www.jianshu.com/p/c9de414221ac?utm_campaign=haruki&utm_content=note&utm_medium= ...

  2. 微信小程序 背景音频播放遇到的深坑

    1.微信前台(聊天页)暂停后回到小程序,再点击播放,播放信息消失,无法续播 ios可以监听到 (onStop已经停止)事件, 安卓无法监听到,只能监听到普通的暂停事件. 2.

  3. Django框架架构总览

    Django框架架构总览 理解Django是如何运作的 条目创建于 2013-08-14     1464 views     服务器君一共花费 15.204 ms 进行了 4 次数据库查询,努力地为 ...

  4. js用逗号分隔字符串,保留双引号中的字符串

    /** * 按逗号分解字符串, 引号中的逗号要保留, 并去除空值. * 方案: 1.将引号中都逗号替换为一个字符串中没有的符号; 2. 用split分解; 3. 去掉空值; 4. 还原引号中的逗号. ...

  5. prefetches

    用于设置预请求的所有url的列表,该部分URL,会在进入小程序后自动发起请求(优于开发者代码加载).当开发者再次发起request请求时可以增加cache参数,如果配置的prefetch请求已返回,则 ...

  6. SPOJ 694 || 705 Distinct Substrings ( 后缀数组 && 不同子串的个数 )

    题意 : 对于给出的串,输出其不同长度的子串的种类数 分析 : 有一个事实就是每一个子串必定是某一个后缀的前缀,换句话说就是每一个后缀的的每一个前缀都代表着一个子串,那么如何在这么多子串or后缀的前缀 ...

  7. Java——方法重写

  8. [CSP-S模拟测试]:随(快速幂+数学)

    题目描述 给出$n$个正整数$a_1,a_2...a_n$和一个质数mod.一个变量$x$初始为$1$.进行$m$次操作.每次在$n$个数中随机选一个$a_i$,然后$x=x\times a_i$.问 ...

  9. CTO爆料:2019程序员最需要了解的行业前沿技术是什么?

    安森,个推CTO 毕业于浙江大学,现全面负责个推技术选型.研发创新.运维管理等工作,已带领团队开发出针对移动互联网.金融风控等行业的多项前沿数据智能解决方案. 曾任MSN中国首席架构师,拥有十余年资深 ...

  10. Zipf's law

    w https://www.bing.com/knows/search?q=马太效应&mkt=zh-cn&FORM=BKACAI 马太效应(Matthew Effect),指强者愈强. ...