题目链接 POJ-3608 Bridge Across Islands

题意

依次按逆时针方向给出凸包,在两个凸包小岛之间造桥,求最小距离。

题解

旋转卡壳的应用之一:求两凸包的最近距离。



找到凸包 p 的 y 值最小点 yminP 和 q 的 y 值最大点ymaxQ,然后分别做切线如图。

那么\(AC\times AD> AC\times AB\)则说明B还不是离AC最近的点,所以++ymaxQ。

否则用 \(AC\) 和 \(BD\) 两个线段的距离更新最近距离,并且++yminP,即考察P的下一条边。

代码

#include <cstdio>
#include <cmath>
#include <algorithm>
#define sqr(x) (x)*(x)
#define N 50001
#define EPS (1e-8)
#define PI acos(-1.0)
#define INF (1e99)
using namespace std;
int sgn(double x) {
if(fabs(x) < EPS)return 0;
return (x < 0)?-1:1;
}
struct Point {
double x,y;
Point(double _x=0,double _y=0):x(_x), y(_y){}
Point operator -(const Point &b)const {
return Point(x - b.x,y - b.y);
}
Point operator +(const Point &b)const {
return Point(x + b.x,y + b.y);
}
double operator ^(const Point &b)const {
return x*b.y - y*b.x;
}
double operator *(const Point &b)const {
return x*b.x + y*b.y;
}
void in(){
scanf("%lf%lf",&x,&y);
}
};
double dis2(Point a,Point b){
return sqr(a-b);
}
double dist(Point a,Point b){
return sqrt(dis2(a,b));
}
struct Line {
Point s,e;
Line(){}
Line(Point _s,Point _e):s(_s),e(_e) {}
};
double xmult(Point a,Point b,Point o){
return (a-o)^(b-o);
}
double mult(Point a, Point b, Point o){
return (a-o)*(b-o);
}
double disToSeg(Point P,Line L){
if(!sgn(dis2(L.s,L.e)))
return dist(L.s,P);
if(sgn(mult(P,L.e,L.s))<0)return dist(L.s,P);
if(sgn(mult(P,L.s,L.e))<0)return dist(L.e,P);
return fabs(xmult(P,L.s,L.e))/dist(L.s,L.e);
}
double segToSeg(Line l1,Line l2){
return min(min(disToSeg(l1.s,l2),disToSeg(l1.e,l2)),min(disToSeg(l2.s,l1),disToSeg(l2.e,l1)));
}
Point p[N],q[N];
int n,m;
double qiake(){
int yminp=0,ymaxq=0;
for(int i=1;i<n;++i)
if(p[i].y<p[yminp].y)
yminp=i;
for(int i=1;i<m;++i)
if(q[i].y>q[ymaxq].y)
ymaxq=i;
p[n]=p[0];
q[m]=q[0];
double tmp,ans=INF;
for(int i=0;i<n;++i){
while(tmp=sgn(xmult(p[yminp+1],q[ymaxq+1],p[yminp])
-xmult(p[yminp+1],q[ymaxq],p[yminp]))>0)
ymaxq=(ymaxq+1)%m;
ans=min(ans,segToSeg(Line(p[yminp],p[yminp+1]),Line(q[ymaxq],q[ymaxq+1])));
yminp=(yminp+1)%n;
}
return ans;
}
int main(){
while(~scanf("%d%d",&n,&m)&&n&&m){
for(int i=0;i<n;++i)
p[i].in();
for(int i=0;i<m;++i)
q[i].in();
printf("%f\n",qiake());
}
return 0;
}

「POJ-3608」Bridge Across Islands (旋转卡壳--求两凸包距离)的更多相关文章

  1. POJ3608(旋转卡壳--求两凸包的最近点对距离)

    题目:Bridge Across Islands 分析:以下内容来自:http://blog.csdn.net/acmaker/article/details/3178696 考虑如下的算法, 算法的 ...

  2. POJ 3608 Bridge Across Islands(旋转卡壳,两凸包最短距离)

    Bridge Across Islands Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7202   Accepted:  ...

  3. POJ 3608 Bridge Across Islands [旋转卡壳]

    Bridge Across Islands Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 10455   Accepted: ...

  4. poj 3608 旋转卡壳求不相交凸包最近距离;

    题目链接:http://poj.org/problem?id=3608 #include<cstdio> #include<cstring> #include<cmath ...

  5. poj 3608(旋转卡壳求解两凸包之间的最短距离)

    Bridge Across Islands Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 9768   Accepted: ...

  6. 旋转卡壳求两个凸包最近距离poj3608

    #include <iostream> #include <cmath> #include <vector> #include <string.h> # ...

  7. POJ2187 旋转卡壳 求最长直径

    给定平面上的一些散点集,求最远两点距离的平方值. 题解: 旋转卡壳求出凸包,然后根据单调性,求出最远两点的最大距离 #pragma GCC optimize(2) #pragma G++ optimi ...

  8. 「POJ 3666」Making the Grade 题解(两种做法)

    0前言 感谢yxy童鞋的dp及暴力做法! 1 算法标签 优先队列.dp动态规划+滚动数组优化 2 题目难度 提高/提高+ CF rating:2300 3 题面 「POJ 3666」Making th ...

  9. POJ 2187 Beauty Contest【旋转卡壳求凸包直径】

    链接: http://poj.org/problem?id=2187 http://acm.hust.edu.cn/vjudge/contest/view.action?cid=22013#probl ...

随机推荐

  1. Codeforces Round #521 (Div. 3)

    B 题过的有些牵强,浪费了很多时间,这种题一定想好思路和边界条件再打,争取一发过.  D 题最开始读错题,后面最后发现可以重复感觉就没法做了,现在想来,数据量大,但是数据范围小枚举不行,二分还是可以的 ...

  2. [Offer收割]编程练习赛97

    链接 [https://hihocoder.com/contest/offers97/problems] 题意 题目1 : 放置矩形 时间限制:10000ms 单点时限:1000ms 内存限制:256 ...

  3. c++入门之类继承初步

    继承是面向对象的一种很重要的特性,先来复习基类的基本知识: 先上一段代码: # ifndef TABLE00_H # define TABLE00_H # include "string&q ...

  4. 使用C# HttpWebRequest进行多线程网页提交。Async httpclient/HttpWebRequest实现批量任务的发布及异步提交和超时取消

    使用线程池并发处理request请求及错误重试,使用委托处理UI界面输出. http://www.cnblogs.com/Charltsing/p/httpwebrequest.html for (i ...

  5. Python的socket模块与交互式指令

    socket简介 在编程的过程中,我们需要使用网络编程,这时我们不得不和网络通信的底层基础打交道了.我们必须让自己传输的数据符合网络通信的基本协议,即TCP/IP协议,但是网络通信协议本身很复杂.我们 ...

  6. Linux sed使用方法

    目录 sed处理流程 测试数据 sed命令格式 sed命令行格式 行定位 定位1行 定位区间行(多行) 定位某一行之外的行 定位有跨度的行 操作命令 -a (新增行) -i(插入行) -c(替代行) ...

  7. python3 打开页面后多窗口处理三种方法

    多窗口处理三种方法 导包,实例化浏览器from selenium import webdriver fx=webdriver.Firefox()方法一fx.switch_to.window(fx.wi ...

  8. Yii的操作提示框

    效果如图 HTML + CSS<style> div.error{ background: #FFE0E0; border: 2px solid #FFA0A0; padding: 10p ...

  9. Farm Irrigation

    题目:Farm Irrigation 题目链接:http://210.34.193.66:8080/vj/Problem.jsp?pid=1494 题目思路:并查集 #include<stdio ...

  10. ajax设置默认值ajaxSetup()方法

    $(function(){ //设置全局 jQuery Ajax全局参数 $.ajaxSetup({ type:"POST", async:false, cache:false, ...