Byteotia岛屿是一个凸多边形。城市全都在海岸上。按顺时针编号1到n。任意两个城市之间都有一条笔直的道路相连。道路相交处可以自由穿行。有一些道路被游击队控制了,不能走,但是可以经过这条道路与未被控制的道路的交点。问从城市1到n的最短距离。
Byteotia岛屿是一个凸多边形。城市全都在海岸上。按顺时针编号1到n。任意两个城市之间都有一条笔直的道路相连。道路相交处可以自由穿行。有一些道路被游击队控制了,不能走,但是可以经过这条道路与未被控制的道路的交点。问从城市1到n的最短距离。
第一行正整数n m表示城市数和被控制的岛屿数(3≤n≤10^5 1≤m≤10^6)接下来n行每行两个整数x y表示每个城市的坐标。(|x|,|y|≤10^6)接下来m行描述一条不能走的道路(起点和终点)。数据保证有解。
输出一个实数,最短距离,误差10^-5以内均算正确。
鸣谢 vfleaking
半平面交思路好题
因为1-n是顺时针排列,所以1-n的最短路径就是所有可以使用的线段的半平面交的长度。
然而这样也是不行的因为边最多会有O(n^2)条。
每个点出发的点,一定是越靠后越好,所以只要保存最靠后那条,这样边就只有O(n)条了。
#include<iostream> #include<cstdlib> #include<cmath> #include<cstdio> #include<cstring> #include<algorithm> #define F(i,j,n) for(int i=j;i<=n;i++) #define D(i,j,n) for(int i=j;i>=n;i--) #define ll long long #define maxn 100005 #define maxm 1000005 #define eps 1e-8 using namespace std; int n,m,cnt,tot; double ans; struct P { double x,y; P(double xx=0,double yy=0){x=xx;y=yy;} friend P operator -(P a,P b){return P(a.x-b.x,a.y-b.y);} friend double operator *(P a,P b){return a.x*b.y-a.y*b.x;} }p[maxn],a[maxn]; struct L { P a,b;double angle; friend bool operator <(L a,L b) { if (fabs(a.angle-b.angle)<=eps) return (a.b-a.a)*(b.b-a.a)>0; else return a.angle<b.angle; } }l[maxn],q[maxn]; struct edge { int x,y; friend bool operator <(edge a,edge b){return a.x==b.x?a.y>b.y:a.x<b.x;} }e[maxm]; inline int read() { int x=0,f=1;char ch=getchar(); while (ch<'0'||ch>'9'){if (ch=='-') f=-1;ch=getchar();} while (ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();} return x*f; } inline double dis(P a,P b) { return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y)); } inline P inter(L l1,L l2) { double k1=(l2.b-l1.a)*(l1.b-l1.a),k2=(l1.b-l1.a)*(l2.a-l1.a),t=k1/(k1+k2); return P(l2.b.x+(l2.a.x-l2.b.x)*t,l2.b.y+(l2.a.y-l2.b.y)*t); } inline bool judge(L l1,L l2,L t) { P p=inter(l1,l2); return (t.b-t.a)*(p-t.a)<-eps; } inline void hpi() { F(i,1,cnt) l[i].angle=atan2(l[i].b.y-l[i].a.y,l[i].b.x-l[i].a.x); sort(l+1,l+cnt+1); int head=1,tail=0;tot=1; F(i,2,cnt) { if (fabs(l[i].angle-l[i-1].angle)>=eps) tot++; l[tot]=l[i]; } cnt=tot; q[++tail]=l[1];q[++tail]=l[2]; F(i,3,cnt) { while (head<tail&&judge(q[tail],q[tail-1],l[i])) tail--; while (head<tail&&judge(q[head],q[head+1],l[i])) head++; q[++tail]=l[i]; } while (head<tail&&judge(q[tail],q[tail-1],q[head])) tail--; tot=0; F(i,head,tail-1) a[++tot]=inter(q[i],q[i+1]); a[++tot]=inter(q[head],q[tail]); } int main() { n=read();m=read(); F(i,1,n) p[i].x=read(),p[i].y=read(); F(i,1,m) { e[i].x=read();e[i].y=read(); if (e[i].x>e[i].y) swap(e[i].x,e[i].y); } sort(e+1,e+m+1); if (e[1].x!=1||e[1].y!=n) { printf("%.10lf\n",dis(p[1],p[n])); return 0; } int t=1; F(i,1,n-1) { int j=n; while (t<=m&&e[t].x==i&&e[t].y==j&&j>i) j--,t++; if (j>i) l[++cnt]=(L){p[j],p[i]}; while (t<=m&&e[t].x==i) t++; } l[++cnt]=(L){p[1],p[n]}; hpi(); a[0]=a[tot]; F(i,1,tot) ans+=dis(a[i],a[i-1]); ans-=dis(p[1],p[n]); printf("%.10lf\n",ans); return 0; }