Problem Description
虽然草儿是个路痴(就是在杭电待了一年多,居然还会在校园里迷路的人,汗~),但是草儿仍然很喜欢旅行,因为在旅途中 会遇见很多人(白马王子,^0^),很多事,还能丰富自己的阅历,还可以看美丽的风景……草儿想去很多地方,她想要去东京铁塔看夜景,去威尼斯看电影,去阳明山上看海芋,去纽约纯粹看雪景,去巴黎喝咖啡写信,去北京探望孟姜女……眼看寒假就快到了,这么一大段时间,可不能浪费啊,一定要给自己好好的放个假,可是也不能荒废了训练啊,所以草儿决定在要在最短的时间去一个自己想去的地方!因为草儿的家在一个小镇上,没有火车经过,所以她只能去邻近的城市坐火车(好可怜啊~)。
 
Input
输入数据有多组,每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个,草儿想去的地方有D个;
接着有T行,每行有三个整数a,b,time,表示a,b城市之间的车程是time小时;(1=<(a,b)<=1000;a,b 之间可能有多条路)
接着的第T+1行有S个数,表示和草儿家相连的城市;
接着的第T+2行有D个数,表示草儿想去地方。
 
Output
输出草儿能去某个喜欢的城市的最短时间。

如题 , 此题是一道明显的最短路问题,可以用dijkstra 和 spfa 等解决 。一般的做法很容易想到,就是求出所有出发的站 到所有 终点站的 最短路径中的最小值 ,这样就重复多次调用dijkstra 或 spfa , 但如果运用一些技巧就可大大优化 , 题目中a , b 均是大于1的,所以可以在设一个点作为草儿的家的位置且该点的序号为 0  , 只要把该点与所有始发站之间均建立一条边且距离为0 ,那么只要以点0 为源点 调用一次dijkstra 或 spfa 就可以了。我用的是spfa ,哎,这道题WA了无数次,就是找不到错在哪里,最后突然间想到,可能有的目的地是孤立的点(在这里指草儿无法到达的点 , 即前面未出现过的点)。请看代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<algorithm>
#include<queue>
#include<cmath>
#include<map>
using namespace std ;
const int MAXN = 1005 ;
const int INF = 0x7fffffff ;
int vis[MAXN] ; // 标记数组,确认城市是否出现过
struct Node
{
int adj ;
int dist ;
Node * next ;
} ;
Node * vert[MAXN] ;
queue <int> q ;
int m , st , dt ;
int dest[MAXN] ;
int ss[MAXN] ; // 记录出发站的城市数目
int dd[MAXN] ; // 记录终点站的城市数目
int dis[MAXN] ;
int inq[MAXN] ;
int sumc ; // 记录出现的不同的城市数目
void spfa(int v0)
{
Node * p ;
int i ;
for(i = 0 ; i <= sumc ; i ++)
{
dis[dest[i]] = INF ;
}
dis[0] = 0 ;
while (!q.empty())
{
q.pop() ;
}
q.push(v0) ;
inq[v0] ++ ;
while (!q.empty())
{
int tmp = q.front() ;
q.pop() ;
inq[tmp] -- ;
p = vert[tmp] ;
while (p != NULL)
{
int td = p -> dist ;
int tadj = p -> adj ;
if(td + dis[tmp] < dis[tadj])
{
dis[tadj] = td + dis[tmp] ;
if(inq[tadj] == 0)
{
inq[tadj] ++ ;
q.push(tadj) ;
}
}
p = p -> next ;
}
}
}
void dele() // 删除邻接表
{
Node * p ;
int i ;
for(i = 0 ; i <= sumc ; i ++)
{
if(i == 0)
p = vert[0] ;
else
p = vert[dest[i]] ;
while (p != NULL)
{
vert[dest[i]] = p -> next ;
delete p ;
p = vert[dest[i]] ;
}
}
}
int main()
{
while (scanf("%d%d%d" , &m , &st , &dt) != EOF)
{
memset(vis , 0 , sizeof(vis)) ;
memset(vert , 0 , sizeof(vert)) ;
memset(dest , 0 , sizeof(dest)) ;
memset(dis , 0 , sizeof(dis)) ;
memset(inq , 0 , sizeof(inq)) ;
int i ;
sumc = 0 ;
Node * p ;
for(i = 0 ; i < m ; i ++)
{
int a , b , w ;
cin >> a >> b >> w ;
if(!vis[a])
{
vis[a] = 1 ;
sumc ++ ;
dest[sumc] = a ;
}
if(!vis[b])
{
vis[b] = 1 ;
sumc ++ ;
dest[sumc] = b ;
}
p = new Node ;
p -> adj = b ;
p -> dist = w ;
p -> next = vert[a] ;
vert[a] = p ; p = new Node ;
p -> adj = a ;
p -> dist = w ;
p -> next = vert[b] ;
vert[b] = p ;
}
for( i = 0 ; i < st ; i ++)
{
scanf("%d" , & ss[i]) ;
if(!vis[ss[i]]) // 这里也不要忘记判断
{
vis[ss[i]] = 1 ;
sumc ++ ;
dest[sumc] = ss[i] ;
}
p = new Node ;
p -> adj = ss[i] ;
p -> dist = 0 ;
p -> next = vert[0] ;
vert[0] = p ; p = new Node ;
p -> adj = 0 ;
p -> dist = 0 ;
p -> next = vert[ss[i]] ;
vert[ss[i]] = p ;
}
for( i = 0 ; i < dt ; i ++)
{
scanf("%d" , & dd[i]) ;
if(!vis[dd[i]]) // 这里也不要忘记判断,终点站的城市可能第一次出现
{
vis[dd[i]] = 1 ;
sumc ++ ;
dest[sumc] = dd[i] ;
}
}
spfa(0) ;
int min = INF ;
for( i = 0 ; i < dt ; i ++)
{
if(min > dis[dd[i]])
{
min = dis[dd[i]] ;
}
}
printf("%d\n" , min) ;
dele() ;
}
return 0 ;
}

HDU 2066 一个人的旅行 - from lanshui_Yang的更多相关文章

  1. hdu 2066 一个人的旅行

    题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=2066 一个人的旅行 Description 虽然草儿是个路痴(就是在杭电待了一年多,居然还会在校园里迷 ...

  2. hdu 2066 一个人的旅行 Dijkstra

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2066 题意分析:以草儿家为原点,给出城市间相互抵达的时间,给出草儿想去的城市,求最短时间.典型的单源最 ...

  3. hdu 2066 一个人的旅行(最短路问题)

    最短路································· 类似的问题还有好多不会!慢慢学吧!!!!. 进步,哪怕每天一点也行! (恋爱不是小事,确实小事的积累!(听着酷狗音乐台说的,很 ...

  4. HDU 2066 一个人的旅行(dijkstra水题+判重边)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2066 题目大意:输入数据有多组,每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有 ...

  5. hdu 2066 一个人的旅行 解题报告

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2066 题目意思:给出T条路,和草儿家相邻的城市编号,以及草儿想去的地方的编号.问从草儿家到达草儿想去的 ...

  6. hdu - 2066 一个人的旅行(基础最短路)

    http://acm.hdu.edu.cn/showproblem.php?pid=2066 把与草儿相连的城市最短距离置为0,然后进行dijkstra,在t个城市里找出距离最近的一个即可. #inc ...

  7. HDU 2066 一个人的旅行(单源最短路SPFA)

    Description 虽然草儿是个路痴(就是在杭电待了一年多,居然还会在校园里迷路的人,汗~),但是草儿仍然很喜欢旅行,因为在旅途中 会遇见很多人(白马王子,^0^),很多事,还能丰富自己的阅历,还 ...

  8. hdu 2066 一个人的旅行 最短路径

    一个人的旅行 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  9. HDU 2066 一个人的旅行 (Dijkstra算法)

    一个人的旅行 Time Limit : 1000/1000ms (Java/Other)   Memory Limit : 32768/32768K (Java/Other) Total Submis ...

随机推荐

  1. 左侧菜单 z

    Dev 的tabControl

  2. nsDATA 转结构体

    很多时候需要将c,c++形式的struct转换为  NSData来处理.但是怎么转换呢? 假设有这么一个结构体: struct   MYINFO { int   a; long  b; char  c ...

  3. php 页面参数过多时自动拼接get参数的函数

    function getUri($query){ $request_uri = $_SERVER["REQUEST_URI"]; $url = strstr($request_ur ...

  4. Vs2015 win10虚拟机启动问题:无法设置UDP端口 解决方法 合集(转载)

    刚装的vs2015 社区版 出现这个问题,wp8.1和win10m模拟器都无法启动,找了好久找到的解决方案,放这儿供大家参考,免得大家像我一样走弯路: Windows Phone emulator n ...

  5. 从 Page not found: / 提示说起,我是怎么发现webstrom与myeclipse冲突问题及解决的

    #从 Page not found: / 提示说起,我是怎么发现webstrom与myeclipse冲突问题的 ##  从前面发表了两篇博文,[webstorm+nodejs+JetBrains ID ...

  6. LeetCode(3) - Longest Substring Without Repeating Characters

    这题的题意大概就是给你一个字符串"abcdecde",找到最长的子字符串长度,里面所有的子母都不重复.本例子中最长的满足条件的子字符串就是"abcde",所以应 ...

  7. Android签名用keytool和jarsigner制作apk文件

    生成证书 keytool -genkey -alias aeo_android.keystore -keyalg RSA -validity -keystore aeo_android.keystor ...

  8. 【转】log4j详解及简易搭建

    原文链接:http://www.cnblogs.com/mailingfeng/archive/2011/07/28/2119937.html log4j是一个非常强大的log记录软件. 首先当然是得 ...

  9. java getEnv不区分大小写 getProperty区分大小写

    System.out.println(System.getenv("JAVA_HOME")); System.out.println(System.getenv("Pat ...

  10. [iOS微博项目 - 3.3] - 封装网络请求

    github: https://github.com/hellovoidworld/HVWWeibo   A.封装网络请求 1.需求 为了避免代码冗余和对于AFN框架的多处使用导致耦合性太强,所以把网 ...