bzoj 1185 [HNOI2007]最小矩形覆盖——旋转卡壳
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1185
矩形一定贴着凸包的一条边。不过只是感觉这样。
枚举一条边,对面的点就是正常的旋转卡壳。两边的那个点可以用点积的最小/大来判断,因为是投影。
然后调了一万年。不过好像把精度设成 1e-13 而不是 1e-8 就能过了。
和许多代码对拍,有各种各样的不同。也不知道自己是不是真的对了。
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define db double
using namespace std;
const int N=5e4+;const db eps=1e-,INF=1e18;
int n,tot; db ans;
struct Node{
db x,y;
Node(db a=,db b=):x(a),y(b) {}
bool operator< (const Node &b)const {return x<b.x||(x==b.x&&y<b.y);}
Node operator- (const Node &b)const {return Node(x-b.x,y-b.y);}
Node operator+ (const Node &b)const {return Node(x+b.x,y+b.y);}
db operator* (const Node &b)const {return x*b.x+y*b.y;}
Node operator* (const db &b)const {return Node(x*b,y*b);}
Node operator/ (const db &b)const {return Node(x/b,y/b);}
}t[N],a[N],prn[];
db Mx(db a,db b){return a>b?a:b;}
int dcmp(db x){if(x>eps)return ;if(x<-eps)return -;return ;}
db cross(Node u,Node v){return u.x*v.y-u.y*v.x;}
void upd(int &x){x>n?x-=n:;}
db Sqr(db x){return x*x;}
db dist(Node u,Node v){return sqrt(Sqr(u.x-v.x)+Sqr(u.y-v.y));}
int main()
{
scanf("%d",&tot);
for(int i=;i<=tot;i++)scanf("%lf%lf",&t[i].x,&t[i].y);
sort(t+,t+tot+);
for(int i=;i<=tot;i++)
{while(n>&&dcmp(cross(a[n]-t[i],a[n-]-t[i]))>=)n--;a[++n]=t[i];}
for(int i=tot-,lm=n;i;i--)
{while(n>lm&&dcmp(cross(a[n]-t[i],a[n-]-t[i]))>=)n--;a[++n]=t[i];}
a[n]=a[]; n--; ans=INF;
int p0=,cr=,p1=;//out of for()
for(int i=;i<=n;i++)
{
Node d=a[i+]-a[i];
while(dcmp(cross(d,a[cr+]-a[i])-cross(d,a[cr]-a[i]))>)cr++,upd(cr);
while(dcmp((a[p0+]-a[i])*d-(a[p0]-a[i])*d)>=)p0++,upd(p0);//>=?
if(i==)p1=cr;//
while(dcmp((a[p1+]-a[i])*d-(a[p1]-a[i])*d)<=)p1++,upd(p1); db len=dist(a[i+],a[i]);
db s1=(a[p0]-a[i])*d/len, s2=(a[p1]-a[i])*d/len;
db hi=cross(d,a[cr]-a[i])/len, ara=(s1-s2)*hi;//not /(2*len)
if(dcmp(ara-ans)>=)continue; ans=ara;
// if(ara>=ans)continue; ans=ara;
Node x1=d/len*(s1-s2);
Node x2=d/len*hi; swap(x2.x,x2.y); x2.x=-x2.x;//*i
prn[]=a[i]+d/len*s2; prn[]=prn[]+x1;
prn[]=prn[]+x2; prn[]=prn[]+x2;
}
printf("%.5f\n",ans);int id=;
for(int i=;i<=;i++)
{
Node u=prn[i],v=prn[id];
if(dcmp(u.y-v.y)<||(!dcmp(u.y-v.y)&&dcmp(u.x-v.x)<))id=i;
}
for(int j=;j<=;j++,id++,id>?id=:)
{
if(!dcmp(prn[id].x))prn[id].x=;
if(!dcmp(prn[id].y))prn[id].y=;//avoid -0
printf("%.5f %.5f\n",prn[id].x,prn[id].y);
}
return ;
}
bzoj 1185 [HNOI2007]最小矩形覆盖——旋转卡壳的更多相关文章
- 洛谷 P3187 BZOJ 1185 [HNOI2007]最小矩形覆盖 (旋转卡壳)
题目链接: 洛谷 P3187 [HNOI2007]最小矩形覆盖 BZOJ 1185: [HNOI2007]最小矩形覆盖 Description 给定一些点的坐标,要求求能够覆盖所有点的最小面积的矩形, ...
- BZOJ 1185: [HNOI2007]最小矩形覆盖 [旋转卡壳]
1185: [HNOI2007]最小矩形覆盖 Time Limit: 10 Sec Memory Limit: 162 MBSec Special JudgeSubmit: 1435 Solve ...
- BZOJ 1185: [HNOI2007]最小矩形覆盖-旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标-备忘板子
来源:旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标 BZOJ又崩了,直接贴一下人家的代码. 代码: #include"stdio.h" #include"str ...
- BZOJ:1185: [HNOI2007]最小矩形覆盖
1185: [HNOI2007]最小矩形覆盖 这计算几何……果然很烦…… 发现自己不会旋转卡壳,补了下,然后发现求凸包也不会…… 凸包:找一个最左下的点,其他点按照与它连边的夹角排序,然后维护一个栈用 ...
- bzoj1185 [HNOI2007]最小矩形覆盖 旋转卡壳求凸包
[HNOI2007]最小矩形覆盖 Time Limit: 10 Sec Memory Limit: 162 MBSec Special JudgeSubmit: 2081 Solved: 920 ...
- bzoj 1185 [HNOI2007]最小矩形覆盖 凸包+旋转卡壳
题目大意 用最小矩形覆盖平面上所有的点 分析 有一结论:最小矩形中有一条边在凸包的边上,不然可以旋转一个角度让面积变小 简略证明 我们逆时针枚举一条边 用旋转卡壳维护此时最左,最右,最上的点 注意 注 ...
- BZOJ 1185 [HNOI2007]最小矩形覆盖:凸包 + 旋转卡壳
题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1185 题意: 给出二维平面上的n个点,问你将所有点覆盖的最小矩形面积. 题解: 先找出凸 ...
- ●BZOJ 1185 [HNOI2007]最小矩形覆盖
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=1185 题解: 计算几何,凸包,旋转卡壳 结论:矩形的某一条边在凸包的一条边所在的直线上. ( ...
- 【bzoj1185】[HNOI2007]最小矩形覆盖 (旋转卡壳)
给你一些点,让你用最小的矩形覆盖这些点 首先有一个结论,矩形的一条边一定在凸包上!!! 枚举凸包上的边 用旋转卡壳在凸包上找矩形另外三点... 注意精度问题 #include<cstdio> ...
随机推荐
- Gruntjs提高生产力(四)
思考: 1.grunt以工程为单位安装插件? 如果有新工程就要重新安装插件或者把安装好的插件拷贝过去.这样很麻烦,解决方案是需要用grunt的项目统一放在grunt项目中. 2.每次需要针对项目编写g ...
- umilit 修改 linux 最多可打开文件数
ulimit -n 修改 临时修改: ulimit -SHn 65535 永久修改: echo '* - nofile 65535' >> /etc/security/li ...
- Post with HttpClient4
转载:http://www.cnblogs.com/luxiaoxun/p/6165237.html 作者:阿凡卢 出处:http://www.cnblogs.com/luxiaoxun/ HttpC ...
- How to create Oracle ASM devices using device-mapper multipath devices in Red Hat Enterprise Linux 6
How to create Oracle ASM devices using device-mapper multipath devices in Red Hat Enterprise Linux 6 ...
- div居中和table居中,jQuery获取下拉列表值
一.div居中 margin-left: auto;margin-right: auto; <div style="width:960px ; margin-left: auto;m ...
- qml 关于鼠标穿透的问题
最近在开发过程中,遇到了鼠标穿透的问题.结合网上给予的方法,都试了一圈,在这里总结一下: import QtQuick 2.9import QtQuick.Window 2.2import QtQui ...
- 《Effective C++》第2章 构造/析构/赋值运算(2)-读书笔记
章节回顾: <Effective C++>第1章 让自己习惯C++-读书笔记 <Effective C++>第2章 构造/析构/赋值运算(1)-读书笔记 <Effecti ...
- Selenium实现右键保存图片(Java)
1.代码 public class SaveImage extends TestCase { private WebDriver driver; private Actions action; pri ...
- Makefile特殊标签
http://www.gnu.org/software/make/manual/html_node/Special-Targets.html
- C++实现线程同步的几种方式
线程同步是指同一进程中的多个线程互相协调工作从而达到一致性.之所以需要线程同步,是因为多个线程同时对一个数据对象进行修改操作时,可能会对数据造成破坏,下面是多个线程同时修改同一数据造成破坏的例子: # ...