BZOJ2596 : [Wc2007]疯狂赛车

根据光路最快原理以及斯涅尔定律,可以得到从定点$P$进入某条直线的最佳入射角。

求出每个端点到每条线段的最佳点,建图求最短路即可。

时间复杂度$O(n^2\log n)$。

 

#include<cstdio>
#include<cmath>
#include<algorithm>
#include<vector>
#include<queue>
using namespace std;
typedef pair<double,int>PI;
const int N=2100000,M=9100000;
const double eps=1e-6,inf=1e100;
int n,cnt,i,j,g[N],v[M],nxt[M],ed;double w[M],d[N],va,vb,si[2],co[2];
priority_queue<PI,vector<PI>,greater<PI> >q;
inline void add(int x,int y,double z){v[++ed]=y;w[ed]=z;nxt[ed]=g[x];g[x]=ed;}
inline void add2(int x,int y,double z){add(x,y,z),add(y,x,z);}
inline void ext(int x,double y){if(y+eps<d[x])q.push(PI(d[x]=y,x));}
inline int sgn(double x){
  if(x>eps)return 1;
  if(x<-eps)return -1;
  return 0;
}
struct P{
  double x,y;
  P(){}
  P(double _x,double _y){x=_x,y=_y;}
  P operator+(P b){return P(x+b.x,y+b.y);}
  P operator-(P b){return P(x-b.x,y-b.y);}
  P operator*(double b){return P(x*b,y*b);}
  P operator/(double b){return P(x/b,y/b);}
  double operator*(P b){return x*b.x+y*b.y;}
  bool operator==(P b){return !sgn(x-b.x)&&!sgn(y-b.y);}
  double len(){return hypot(x,y);}
  P rotate(double s,double c){return P(x*c-y*s,x*s+y*c);}
  P rot90(){return P(-y,x);}
}a[1010];
struct E{
  double x;int y;
  E(){}
  E(double _x,int _y){x=_x,y=_y;}
}e[2010];
inline bool cmp(const E&a,const E&b){return a.x<b.x;}
inline double cross(P a,P b){return a.x*b.y-a.y*b.x;}
inline bool point_on_segment(P p,P a,P b){
  return sgn(cross(b-a,p-a))==0&&sgn((p-a)*(p-b))<=0;
}
inline P line_intersection(P a,P b,P p,P q){
  double U=cross(p-a,q-p),D=cross(b-a,q-p);
  return a+(b-a)*(U/D);
}
inline void work(int st,int en){
  int i,j,m=2;
  P A=a[st],B=a[en],C=(B-A).rot90();
  e[1]=E(0,st),e[2]=E(C.len(),en);
  for(i=0;i<=n;i++)if(i!=st&&i!=en)for(j=0;j<2;j++){
    P D=line_intersection(A,B,a[i],a[i]+C.rotate(si[j],co[j]));
    if(D==A||D==B)continue;
    if(!point_on_segment(D,A,B))continue;
    cnt++;
    add2(i,cnt,(a[i]-D).len()/vb);
    e[++m]=E((D-A).len(),cnt);
  }
  sort(e+1,e+m+1,cmp);
  for(i=1;i<m;i++)add2(e[i].y,e[i+1].y,(e[i+1].x-e[i].x)/va);
}
int main(){
  scanf("%d%lf%lf",&n,&va,&vb);
  for(i=1;i<=n;i++)scanf("%lf%lf",&a[i].x,&a[i].y);
  if(sgn(va-vb)<=0)return printf("%.10f",(a[0]-a[n]).len()/vb),0;
  si[0]=vb/va;
  co[0]=sqrt(1.0-si[0]*si[0]);
  si[1]=-si[0];
  co[1]=co[0];
  cnt=n;
  for(i=0;i<=n;i++)for(j=0;j<i;j++)add2(i,j,(a[i]-a[j]).len()/vb);
  for(i=0;i<n;i++)work(i,i+1);
  for(i=0;i<=cnt;i++)d[i]=inf;
  ext(0,0);
  while(!q.empty()){
    PI t=q.top();q.pop();
    if(t.first-eps>d[t.second])continue;
    for(i=g[t.second];i;i=nxt[i])ext(v[i],t.first+w[i]);
  }
  return printf("%.10f",d[n]),0;
}

  

posted @ 2017-09-14 01:48  Claris  阅读(579)  评论(0编辑  收藏  举报