POJ 2502 Subway-经过预处理的最短路
Description
You walk at a speed of 10 km/h. The subway travels at 40 km/h. Assume that you are lucky, and whenever you arrive at a subway station, a train is there that you can board immediately. You may get on and off the subway any number of times, and you may switch between different subway lines if you wish. All subway lines go in both directions.
Input
Output
Sample Input
0 0 10000 1000
0 200 5000 200 7000 200 -1 -1
2000 600 5000 600 10000 600 -1 -1
Sample Output
21
Source
这道题的思路简单来说就是将所有地铁站转换成点,然后预处理所有点间的边权(即将距离转换成时间),然后用最短路算法(例如SPFA)求出起点和目标点的距离。
第三:
现在还没解决的问题,如果两条地铁线交叉,那么赋权就又有问题了,所以我希望可以用坐标来表示点,但是由于坐标可能给的很大,所以一直不知道怎么处理。
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#define ll long long
using namespace std;
int read()
{
int x=,y=;
char ch=getchar();
while(ch<''||ch>'')
{
if(ch=='-')
y=-;
ch=getchar();
}
while(ch>=''&&ch<='')
{
x=x*+ch-'';
ch=getchar();
}
return x*y;
}
int abs(int x)
{
if(x<)
return -x;
else
return x;
}
int change(double x)
{
int r=(int)x;
if(x-r<0.5)
return r;
else
return r+;
}
double way(int x1,int y1,int x2,int y2)
{
int r1=abs(x1-x2),r2=abs(y1-y2);
return sqrt(r1*r1+r2*r2);
}
struct edge
{
int next,to;
double lon;
} e[];
int map[][],num,node[][],head[],cnt,t[],headd,tail=;
double dist[];
bool vis[];
void add(int from,int to,double lon)
{
e[++cnt].lon=lon;
e[cnt].to=to;
e[cnt].next=head[from];
head[from]=cnt;
}
int main()
{
memset(head,-,sizeof(head));
int x1=read(),y1=read(),x2=read(),y2=read(),x,y,p=,l=change(way(x1,y1,x2,y2)/*);
node[++num][]=x1;
node[num][]=y1;
node[++num][]=x2;
node[num][]=y2;
add(,,l);
add(,,l);
while(scanf("%d%d",&x,&y)!=EOF)
{
if(x==-&&y==-)
{
p++;
continue;
}
node[++num][]=x;
node[num][]=y;
node[num][]=p;
for(int i=; i<num; i++)
{
double dis;
if(node[i][]==p)
dis=way(x,node[i][],y,node[i][])/*;
else
dis=way(x,node[i][],y,node[i][])/*;
// printf("x=%d y=%d node[i][0]=%d node[i][1]=%d dis=%f\n",x,y,node[i][0],node[i][1],dis);
add(num,i,dis),add(i,num,dis);
}
}
for(int i=; i<=num; i++)
dist[i]=2e8;
t[]=;
while(headd!=tail)
{
int r=head[t[headd]];
vis[t[headd]]=;
while(r!=-)
{
if(dist[e[r].to]>dist[t[headd]]+e[r].lon)
{
dist[e[r].to]=dist[t[headd]]+e[r].lon;
printf("e[r].lon=%f e[r].to=%d dist=%f %f\n",e[r].lon,e[r].to,dist[t[headd]],dist[e[r].to]);
if(!vis[e[r].to])
{
vis[e[r].to]=;
t[tail++]=e[r].to;
}
}
r=e[r].next;
}
headd++;
}
printf("%f",dist[]);
return ;
} // FOR C.H
附上最新经谭姐启发写的正解方法代码(思路稍后,仍有问题在调试):
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
// 邻接链保存边
int head[],cnt,num;
struct edge
{
int next,to;
double lon;
} e[];
void add(int from,int to,double lon)
{
e[++cnt].lon=lon;
e[cnt].to=to;
e[cnt].next=head[from];
head[from]=cnt;
}
// 绝对值函数
int abs(int x)
{
if(x<)
return -x;
else
return x;
}
// 求两点间距离的函数
double far(int x1,int y1,int x2,int y2)
{
int r1=abs(x1-x2),r2=abs(y1-y2);
return sqrt(r1*r1+r2*r2);
}
// 将double转变成int的函数
int change(double x)
{
int r=(int)x;
if(x-r<0.5)
return r;
else
return r+;
}
int map[][];
// 输入处理,自认为比以前有所进步
void scan()
{
int x,y,p=,s=;
while(scanf("%d%d",&x,&y)!=EOF)
{
if(x==-&&y==-)
{
// 该条地铁线上的点和前后的点连无向快边
for(int i=; i<=s; i++)
{
double f=far(map[num-s+i-][],map[num-s+i-][],map[num-s+i][],map[num-s+i][])/*;
add(num-s+i-,num-s+i,f);
add(num-s+i,num-s+i-,f);
}
s=;
continue;
}
map[++num][]=x;
map[num][]=y;
s++;
}
}
// 每点到起点的距离
double dist[];
// SPFA的队列
int t[],headd,tail=;
bool vis[];
int main()
{
memset(head,-,sizeof(head));
int x1,y1,x2,y2;
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
scan();
// 每个点间连一条慢边,虽然可以优化,不用全部连,但是懒得打了,反正最短路嘛
for(int i=; i<=num; i++)
for(int j=; j<=num; j++)
{
double f=far(map[i][],map[i][],map[j][],map[j][])/*;
add(i,j,f);
}
// 每个点和起点连一条慢边
num++;
for(int i=; i<num; i++)
{
double f=far(map[i][],map[i][],x1,y1)/*;
add(i,num,f);
add(num,i,f);
}
// 除起点外和终点连一条慢边
num++;
for(int i=; i<num-; i++)
{
double f=far(map[i][],map[i][],x2,y2)/*;
add(i,num,f);
add(num,i,f);
}
// SPFA求起点到终点的最短路
t[]=num-;
vis[num-]=;
for(int i=; i<=num; i++)
dist[i]=2e8;
dist[num-]=;
while(headd!=tail)
{
int r=head[t[headd]];
printf("r=%d\n",r);
while(r!=-)
{
if(dist[e[r].to]>dist[t[headd]]+e[r].lon)
{
dist[e[r].to]=dist[t[headd]]+e[r].lon;
printf("dist[e[r].to]=%f\n",dist[e[r].to]);
if(!vis[e[r].to])
{
vis[e[r].to]=;
t[tail++]=e[r].to;
}
}
r=e[r].next;
}
headd++;
}
// 输出答案
printf("%d",change(dist[num]));
return ;
} // FOR C.H
终于迎来了最后一次编辑!
之前的程序有一个小错误找了半天,就是SPFA忘记出队,最后终于找到了!
还有,我还是把每个地铁站的点到其他地铁站的距离优化了一下,少加了几条(其实蛮多)边。
最后的AC代码:
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
// 邻接链保存边
int head[],cnt,num;
struct edge
{
int next,to;
double lon;
} e[];
void add(int from,int to,double lon)
{
e[++cnt].lon=lon;
e[cnt].to=to;
e[cnt].next=head[from];
head[from]=cnt;
}
// 绝对值函数
int abs(int x)
{
if(x<)
return -x;
else
return x;
}
// 求两点间距离的函数
double far(int x1,int y1,int x2,int y2)
{
int r1=abs(x1-x2),r2=abs(y1-y2);
return sqrt(r1*r1+r2*r2);
}
// 四舍五入的函数
int change(double x)
{
int r=(int)x;
if(x-r<0.5)
return r;
else
return r+;
}
// 输入处理,自认为比以前有所进步
int map[][];
bool m[][];
void scan()
{
int x,y,p=,s=;
while(scanf("%d%d",&x,&y)!=EOF)
{
if(x==-&&y==-)
{
// 该条地铁线上的点和前后的点连无向快边
for(int i=; i<=s; i++)
{
double f=far(map[num-s+i-][],map[num-s+i-][],map[num-s+i][],map[num-s+i][])/*;
add(num-s+i-,num-s+i,f);
add(num-s+i,num-s+i-,f);
m[num-s+i-][num-s+i]=;
}
s=;
continue;
}
map[++num][]=x;
map[num][]=y;
s++;
}
}
// 每点到起点的距离
double dist[];
// SPFA的队列
int t[],headd,tail=;
bool vis[];
int main()
{
memset(head,-,sizeof(head));
int x1,y1,x2,y2;
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
scan();
// 每个点间连一条慢边,如果已经连了快边了就不用连
for(int i=; i<=num; i++)
for(int j=; j<=num; j++)
{
if(m[i][j]||m[j][i]||i==j)
continue;
double f=far(map[i][],map[i][],map[j][],map[j][])/*;
add(i,j,f);
}
// 每个点和起点连条慢边
num++;
for(int i=; i<num; i++)
{
double f=far(map[i][],map[i][],x1,y1)/*;
add(i,num,f);
add(num,i,f);
}
// 除起点外和终点连条慢边
num++;
for(int i=; i<num-; i++)
{
double f=far(map[i][],map[i][],x2,y2)/*;
add(i,num,f);
add(num,i,f);
}
// SPFA求起点到终点的最短路
t[]=num-;
vis[num-]=;
for(int i=; i<=num; i++)
dist[i]=2e8;
dist[num-]=;
while(headd!=tail)
{
int r=head[t[headd]];
vis[t[headd]]=;
while(r!=-)
{
if(dist[e[r].to]>dist[t[headd]]+e[r].lon)
{
dist[e[r].to]=dist[t[headd]]+e[r].lon;
if(!vis[e[r].to])
{
vis[e[r].to]=;
t[tail++]=e[r].to;
}
}
r=e[r].next;
}
headd++;
}
// 输出答案
printf("%d",change(dist[num]));
return ;
} // FOR C.H
POJ 2502 Subway-经过预处理的最短路的更多相关文章
- POJ 2502 Subway / NBUT 1440 Subway / SCU 2186 Subway(图论,最短距离)
POJ 2502 Subway / NBUT 1440 Subway / SCU 2186 Subway(图论,最短距离) Description You have just moved from a ...
- POJ 2502 - Subway Dijkstra堆优化试水
做这道题的动机就是想练习一下堆的应用,顺便补一下好久没看的图论算法. Dijkstra算法概述 //从0出发的单源最短路 dis[][] = {INF} ReadMap(dis); for i = 0 ...
- POJ 2502 Subway (最短路)
Subway 题目链接: http://acm.hust.edu.cn/vjudge/contest/122685#problem/L Description You have just moved ...
- POJ 2502 Subway ( 最短路 && 最短路建图 )
题意 : 给出二维平面上的两个点代表起点以及终点,接下来给出若干条地铁线路,除了在地铁线路上行进的速度为 40km/h 其余的点到点间都只能用过步行且其速度为 10km/h ,现问你从起点到终点的最短 ...
- POJ 2502 Subway(迪杰斯特拉)
Subway Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 6692 Accepted: 2177 Descriptio ...
- POJ 2502 Subway (Dijkstra 最短+建设规划)
Subway Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 6689 Accepted: 2176 Descriptio ...
- POJ 2502 Subway
Subway Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 4928 Accepted: 1602 Descriptio ...
- (简单) POJ 2502 Subway,Dijkstra。
Description You have just moved from a quiet Waterloo neighbourhood to a big, noisy city. Instead of ...
- Dijkstra+计算几何 POJ 2502 Subway
题目传送门 题意:列车上行驶40, 其余走路速度10.问从家到学校的最短时间 分析:关键是建图:相邻站点的速度是40,否则都可以走路10的速度.读入数据也很变态. #include <cstdi ...
随机推荐
- struts2.1.6教程十二、总结
本教程对struts2的基本知识进行了一些说明,关于struts2的更多详细内容应参看struts2的官方文档及提供的app实例. 下面对struts2的基本执行流程作一简要说明,此流程说明可以结合官 ...
- javaWeb学习总结(11)- 监听器(Listener)学习
一.监听器介绍 1.1.监听器的概念 监听器是一个专门用于对其他对象身上发生的事件或状态改变进行监听和相应处理的对象,当被监视的对象发生情况时,立即采取相应的行动.监听器其 实就是一个实现特定接口的普 ...
- iOS面试题及答案
设计模式是什么? 你知道哪些设计模式,并简要叙述? 设计模式是一种编码经验,就是用比较成熟的逻辑去处理某一种类型的事情. 1). MVC模式:Model View Control,把模型 视图 控制器 ...
- React,关于redux的一点小见解
最近项目做多页面应用使用到了,react + webpack + redux + antd去构建多页面的应用,本地开发用express去模拟服务端程序(个人觉得可以换成dva).所以在这里吐槽一下我自 ...
- VR全景智慧城市——商家的需求才是全景市场的核心竞争力
消费者视角痛点:比如酒店消费行业,很多消费者在预订酒店过程中,都遇到过这样的场景:网上照片里酒店房间看着宽敞明亮,格调不凡,感觉非常喜欢,等真正推开房门插上房卡一看,却大失所望.在酒店行业,网上照片和 ...
- 【webpack】webpack-dev-server生猛上手——让我们来搭一个webpack的微服务器吧!
[前言]:因为最近在搞百度地图API的时候到了webpack的externals,才发现我之前都只是用webpack做一些搭建完项目后的"收尾工作"--即打包,而没有把它纳入到 ...
- java面试题之int和Integer的区别
int和Integer的区别 1.Integer是int的包装类,int则是java的一种基本数据类型 2.Integer变量必须实例化后才能使用,而int变量不需要 3.Integer实际是对象的引 ...
- 关于ASP.NET WebForm与ASP.NET MVC的比较
WebForm的理解 1. WebForm概念 ASP.NETWebform提供了一个类似于Winform的事件响应GUI模型(event-drivenGUI),隐藏了HTTP.HTML.Java ...
- ap.net core 教程(三) - 新建项目
ASP.NET Core - 新建项目 在这一章,我们将讨论如何在Visual Studio中创建一个新项目. 只要你安装了Visual Studio 2015的.net core工具,您就可以开始构 ...
- Perl格式化输出
Perl格式化输出 问题阐述 有时我们需要大量的重复数据,使用手工易于出错及比较繁琐.抓取特征,可以使用Perl脚本轻松搞定. 输出数据格式 主要特点 随机数生成 格式化输出 序列递增 Perl脚本 ...