博客
关于我
【旋转卡壳】poj 2187 Beauty Contest
阅读量:629 次
发布时间:2019-03-14

本文共 2185 字,大约阅读时间需要 7 分钟。

Link:

旋转卡壳(求平面最远对(O(N)))

先求出其凸包,找凸包所有边的与其他点最大三角形,其点即为边的最远点,
再计算边两端点的与边的最远点距离,取下最大距离。
当我们逆时针枚举边的时候,最远点的变化也是逆时针的,这样就可以不用从头计算最远点,
而可以紧接着上一次的最远点继续计算。于是我们得到了O(n)的算法。

#include 
#include
#include
using namespace std;/*poj 2187题意:给你一些点,找最远的两点题解:旋转卡壳*/const int N = 50010;const double PI = acos(-1.0);struct Point{ double x,y;};bool cmp(Point aa,Point bb){ if(aa.y != bb.y) return aa.y < bb.y; return aa.x < bb.x;} //是否严格左转,共线不算(叉乘) //计算三角形面积(叉乘)double xmult(Point a,Point b,Point c) //(ca)×(cb){ return (a.x-c.x)*(b.y-c.y)-(b.x-c.x)*(a.y-c.y);}Point a[N];int n,cnt,tail;int tmp[N],ans[N];void Graham(){ sort(a,a+n,cmp);// for(int i = 0; i < n; i++)// printf("%d %f %f\n",i,a[i].x,a[i].y); cnt = tail = 0; tmp[tail++] = 0; tmp[tail++] = 1; for(int i = 2; i < n; i++) { while(tail>1 && xmult(a[tmp[tail-1]],a[i],a[tmp[tail-2]])<=0) //是否严格左转 tail--; tmp[tail++] = i; } for(int i = 0; i < tail; i++) ans[cnt++] = tmp[i]; tail = 0; tmp[tail++] = n-1; tmp[tail++] = n-2; for(int i = n-3; i >= 0; i--) { while(tail>1 && xmult(a[tmp[tail-1]],a[i],a[tmp[tail-2]])<=0) //是否严格左转 tail--; tmp[tail++] = i; } cnt--; //去掉重复的点 for(int i = 0; i < tail; i++) ans[cnt++] = tmp[i];// for(int i = 0; i < cnt; i++)// printf("%d ",ans[i]);// puts("");}double dist(Point p1,Point p2){ return sqrt((p1.x-p2.x)*(p1.x-p2.x) + (p1.y-p2.y)*(p1.y-p2.y));}int dist2(Point p1,Point p2){ return (int)(p1.x-p2.x)*(p1.x-p2.x) + (int)(p1.y-p2.y)*(p1.y-p2.y);}int rotating_calipers() //旋转卡壳{ int q=1; //处理到的点 int res = 0; for(int i = 0; i < cnt-1; i++){ while(xmult(a[ans[q+1]],a[ans[i]],a[ans[i+1]]) > xmult(a[ans[q]],a[ans[i]],a[ans[i+1]])) //三角形面积 q = (q+1)%(cnt-1); res = max(res,max(dist2(a[ans[i]],a[ans[q]]),dist2(a[ans[i+1]],a[ans[q]])));// printf("%d %d %d\n",ans[i],ans[i+1],q); } return res;}int main(){ while(~scanf("%d",&n)) { for(int i = 0; i < n; i++) scanf("%lf%lf",&a[i].x,&a[i].y); Graham(); printf("%d\n",rotating_calipers()); } return 0;}

转载地址:http://wiaoz.baihongyu.com/

你可能感兴趣的文章
MYSQL 幻读(Phantom Problem)不可重复读
查看>>
mysql 往字段后面加字符串
查看>>
mysql 快照读 幻读_innodb当前读 与 快照读 and rr级别是否真正避免了幻读
查看>>
MySQL 快速创建千万级测试数据
查看>>
mysql 快速自增假数据, 新增假数据,mysql自增假数据
查看>>
MySql 手动执行主从备份
查看>>
Mysql 批量修改四种方式效率对比(一)
查看>>
Mysql 报错 Field 'id' doesn't have a default value
查看>>
MySQL 报错:Duplicate entry 'xxx' for key 'UNIQ_XXXX'
查看>>
Mysql 拼接多个字段作为查询条件查询方法
查看>>
mysql 排序id_mysql如何按特定id排序
查看>>
Mysql 提示:Communication link failure
查看>>
mysql 插入是否成功_PDO mysql:如何知道插入是否成功
查看>>
Mysql 数据库InnoDB存储引擎中主要组件的刷新清理条件:脏页、RedoLog重做日志、Insert Buffer或ChangeBuffer、Undo Log
查看>>
mysql 数据库中 count(*),count(1),count(列名)区别和效率问题
查看>>
mysql 数据库备份及ibdata1的瘦身
查看>>
MySQL 数据库备份种类以及常用备份工具汇总
查看>>
mysql 数据库存储引擎怎么选择?快来看看性能测试吧
查看>>
MySQL 数据库操作指南:学习如何使用 Python 进行增删改查操作
查看>>
MySQL 数据库的高可用性分析
查看>>