假设S在T左边,那么只能往右或者上下走

f[i]表示S到i点的最短路

f[i]=min(f[j]+dis(i,j)(i能看到j))

判断i能看到j就维护一个上凸壳和一个下凸壳

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

代码写的有点长…

#include<cstdio>
#include<cmath>
#include<algorithm>
#define N 2010
using namespace std;
struct P{int x,y;P(){}P(int _x,int _y){x=_x,y=_y;}P operator-(const P&a){return P(x-a.x,y-a.y);}};
inline int cross(P a,P b){return a.x*b.y-a.y*b.x;}
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));}
int n,i,X1[N],Y1[N],X2[N],Y2[N],xs,ys,xt,yt,st,en;
double v,f[N][4],ans,inf=1e9;
inline void call(int x,int y,int p,double&t){
t=inf;
P o(x,y),up(x,y+1),down(x,y-1),tmp;
for(int i=p-1;i>st;i--){
tmp=P(X2[i],Y2[i]);
if(cross(up-o,tmp-o)>=0){//在up下边或者重合
up=tmp;
if(cross(down-o,tmp-o)<=0)t=min(t,f[i][3]+dis(o,tmp));//在down上边或者重合
}
tmp=P(X2[i],Y1[i]);
if(cross(down-o,tmp-o)<=0){//在down上边或者重合
down=tmp;
if(cross(up-o,tmp-o)>=0)t=min(t,f[i][2]+dis(o,tmp));//在up下边或者重合
}
tmp=P(X1[i],Y2[i]);
if(cross(up-o,tmp-o)>=0){//在up下边或者重合
up=tmp;
if(cross(down-o,tmp-o)<=0)t=min(t,f[i][1]+dis(o,tmp));//在down上边或者重合
}
tmp=P(X1[i],Y1[i]);
if(cross(down-o,tmp-o)<=0){//在down上边或者重合
down=tmp;
if(cross(up-o,tmp-o)>=0)t=min(t,f[i][0]+dis(o,tmp));//在up下边或者重合
}
}
tmp=P(X2[st],Y2[st]);
if(cross(up-o,tmp-o)>=0){//在up下边或者重合
up=tmp;
if(cross(down-o,tmp-o)<=0)t=min(t,f[st][3]+dis(o,tmp));//在down上边或者重合
}
tmp=P(X2[st],Y1[st]);
if(cross(down-o,tmp-o)<=0){//在down上边或者重合
down=tmp;
if(cross(up-o,tmp-o)>=0)t=min(t,f[st][2]+dis(o,tmp));//在up下边或者重合
}
tmp=P(xs,ys);
if(cross(up-o,tmp-o)>=0&&cross(down-o,tmp-o)<=0)t=dis(o,tmp);//在up下边或者重合 在down上边或者重合
}
inline void calr(int x,int y,int p,double&t){
t=inf;
P o(x,y),up(x,y+1),down(x,y-1),tmp;
tmp=P(X1[p],Y2[p]);
if(cross(up-o,tmp-o)>=0){//在up下边或者重合
up=tmp;
if(cross(down-o,tmp-o)<=0)t=min(t,f[p][1]+dis(o,tmp));//在down上边或者重合
}
tmp=P(X1[p],Y1[p]);
if(cross(down-o,tmp-o)<=0){//在down上边或者重合
down=tmp;
if(cross(up-o,tmp-o)>=0)t=min(t,f[p][0]+dis(o,tmp));//在up下边或者重合
}
for(int i=p-1;i>st;i--){
tmp=P(X2[i],Y2[i]);
if(cross(up-o,tmp-o)>=0){//在up下边或者重合
up=tmp;
if(cross(down-o,tmp-o)<=0)t=min(t,f[i][3]+dis(o,tmp));//在down上边或者重合
}
tmp=P(X2[i],Y1[i]);
if(cross(down-o,tmp-o)<=0){//在down上边或者重合
down=tmp;
if(cross(up-o,tmp-o)>=0)t=min(t,f[i][2]+dis(o,tmp));//在up下边或者重合
}
tmp=P(X1[i],Y2[i]);
if(cross(up-o,tmp-o)>=0){//在up下边或者重合
up=tmp;
if(cross(down-o,tmp-o)<=0)t=min(t,f[i][1]+dis(o,tmp));//在down上边或者重合
}
tmp=P(X1[i],Y1[i]);
if(cross(down-o,tmp-o)<=0){//在down上边或者重合
down=tmp;
if(cross(up-o,tmp-o)>=0)t=min(t,f[i][0]+dis(o,tmp));//在up下边或者重合
}
}
tmp=P(X2[st],Y2[st]);
if(cross(up-o,tmp-o)>=0){//在up下边或者重合
up=tmp;
if(cross(down-o,tmp-o)<=0)t=min(t,f[st][3]+dis(o,tmp));//在down上边或者重合
}
tmp=P(X2[st],Y1[st]);
if(cross(down-o,tmp-o)<=0){//在down上边或者重合
down=tmp;
if(cross(up-o,tmp-o)>=0)t=min(t,f[st][2]+dis(o,tmp));//在up下边或者重合
}
tmp=P(xs,ys);
if(cross(up-o,tmp-o)>=0&&cross(down-o,tmp-o)<=0)t=dis(o,tmp);//在up下边或者重合 在down上边或者重合
}
int main(){
scanf("%d",&n);
for(i=1;i<=n;i++)scanf("%d%d%d%d",&X1[i],&Y1[i],&X2[i],&Y2[i]);
scanf("%d%d%d%d",&xs,&ys,&xt,&yt);
if(xs>xt)swap(xs,xt),swap(ys,yt);
scanf("%lf",&v);
for(i=1;i<=n;i++)if(X1[i]<=xs&&xs<=X2[i]&&Y1[i]<=ys&&ys<=Y2[i]){st=i;break;}
for(i=n;i;i--)if(X1[i]<=xt&&xt<=X2[i]&&Y1[i]<=yt&&yt<=Y2[i]){en=i;break;}
f[st][2]=dis(P(xs,ys),P(X2[st],Y1[st]));
f[st][3]=dis(P(xs,ys),P(X2[st],Y2[st]));
for(i=st+1;i<en;i++){
call(X1[i],Y1[i],i,f[i][0]);
call(X1[i],Y2[i],i,f[i][1]);
calr(X2[i],Y1[i],i,f[i][2]);
calr(X2[i],Y2[i],i,f[i][3]);
}
call(X1[en],Y1[en],en,f[en][0]);
call(X1[en],Y2[en],en,f[en][1]);
calr(xt,yt,en,ans);
printf("%.10f",ans/v);
return 0;
}

  

Noi2011 : 智能车比赛的更多相关文章

  1. 2433: [Noi2011]智能车比赛 - BZOJ

    Description 新一届智能车大赛在JL大学开始啦!比赛赛道可以看作是由n个矩形区域拼接而成(如下图所示),每个矩形的边都平行于坐标轴,第i个矩形区域的左下角和右上角坐标分别为(xi,1,yi, ...

  2. [NOI2011]智能车比赛 (计算几何 DAG)

    /* 可以发现, 最优路径上的所有拐点, 基本上都满足一定的性质, 也就是说是在矩形上的拐角处 所以我们可以把他们提出来, 单独判断即可 由于我们提出来的不超过2n + 2个点, 我们将其按照x坐标排 ...

  3. [bzoj2433][Noi2011]智能车比赛

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2433 http://221.192.240.123:8586/JudgeOnline/ ...

  4. 【[NOI2011]智能车比赛】(建图+spfa+坑爹精度)

    过了这题我就想说一声艹,跟这个题死磕了将近6个小时,终于是把这个题死磕出来了.首先看到这个题的第一反应,和当初做过的一个房间最短路比较相似,然后考虑像那个题那样建边,然后跑最短路.(具体建边方法请参考 ...

  5. 【LOJ】#2443. 「NOI2011」智能车比赛

    题解 显然是个\(n^2\)的dp 我们要找每个点不穿过非赛道区域能到达哪些区域的交点 可以通过控制两条向量负责最靠下的上边界,和最靠上的下边界,检查当前点在不在这两条向量之间即可,对于每个点可以\( ...

  6. BZOJ 2433 智能车比赛(计算几何+最短路)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2433 题意:若干个矩形排成一排(同一个x之上最多有一个矩形),矩形i和i+1相邻.给定两 ...

  7. 智能车学习(十五)——K60野火2013版例程

    一.中断函数注册方法: 1.格式: 配置某个功能的中断 注册中断函数 开启中断 2.一个例子 pit_init_ms(PIT0,);//定时中断初始化 set_vector_handler(PIT0_ ...

  8. K60平台智能车开发工作随手记

    (图片仅为示例,并不一定固定为这种造型) 第十二届全国大学生智能汽车竞赛有一个分项是光电四轮车的竞速(任务A),Seven她们组采购到的配件使用了freescale Crotex-M4内核的CPU,T ...

  9. 【sky第二期--PID算法】--【智能车论坛】

    [sky第二期--PID算法] 想学PID的可以来[智能车论坛]这里有我发布的资料http://bbs.tekbots.eefocus.com/forum.php?mod=viewthread& ...

随机推荐

  1. ZeroMQ(java)之Router与Dealer运行原理

    在开始这部分的内容之前,先来看看ZeroMQ中HWM概念---High-Water Marks 当系统的数据量很大,而且发送频率很高的情况下,内存就很重要了,如果处理不好会出现很多问题,例如如下场景: ...

  2. 对比WDCP面板与AMH面板的区别与选择

    转载: http://www.laozuo.org/2760.html | 老左博客 随着VPS主机的性价比提高(其实就是降价)我们很多站长会越来越多的选择使用VPS搭建网站或者运营一些项目,相比较而 ...

  3. Windows下的git配置

    需要的配置: 1.C:\Program Files\Git\etc\git-completion.bash: alias ls='ls --show-control-chars --color=aut ...

  4. Android Studio项目引入外部库注意事项(PullToRefresh)

    Android Studio开发App项目时引入第三方库是个比较麻烦的事情.之前导入Volley就折腾了好久,导入下拉刷新控件PullToRefresh时又碰到了各种问题.在此记录一下,以便查阅. 一 ...

  5. gdalwarp切割tif参数

    可以去gdal官网查询gdalwarp工具的参数,但是具体的还是不知道怎么写,例如内置数据类型-ot 和压缩-co参数. 这里有一个经过雁阵更可以使用的参数 gdalwarp -te lon1 lat ...

  6. August 3rd, 2016, Week 32nd, Wednesday

    I am looking for someone to share in an adventure. 我在找能和我一起分享冒险之旅的人. We are all looking for someone ...

  7. struts标签--logic总结

    1. logic:empty 该标签是用来判断是否为空的.如果为空,该标签体中嵌入的内容就会被处理.该标签用于以下情况: 1)当Java对象为null时: 2)当String对象为"&quo ...

  8. python基础——使用list和tuple

    python基础——使用list和tuple list Python内置的一种数据类型是列表:list.list是一种有序的集合,可以随时添加和删除其中的元素. 比如,列出班里所有同学的名字,就可以用 ...

  9. 前端代理nproxy

    一.场景/用途 前端代理的用途,相信大家都清楚.应用场景很多,如—— . 将线上的静态资源文件(JS.CSS.图片)替换为本地相应的文件,来调试线上(代码都被压缩过)的问题: . 本地开发过程,当后端 ...

  10. 【读书笔记】读《JavaScript设计模式》之适配器模式

    一.定义 适配器模式可用来在现有接口和不兼容的类之间进行匹配.使用这种模式的对象又叫包装器(wrapper),因为它们是在用一个新的接口包装另一个对象.在设计类的时候旺旺会遇到有些接口不能与现有API ...