杭电3790————最短路问题
来源:互联网 发布:新加坡电信网络制式 编辑:程序博客网 时间:2024/06/05 03:35
最短路径问题
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 14059 Accepted Submission(s): 4310
Problem Description
给你n个点,m条无向边,每条边都有长度d和花费p,给你起点s终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。
Input
输入n,m,点的编号是1~n,然后是m行,每行4个数 a,b,d,p,表示a和b之间有一条边,且其长度为d,花费为p。最后一行是两个数 s,t;起点s,终点。n和m为0时输入结束。
(1<n<=1000, 0<m<100000, s != t)
(1<n<=1000, 0<m<100000, s != t)
Output
输出 一行有两个数, 最短距离及其花费。
Sample Input
3 21 2 5 62 3 4 51 30 0
Sample Output
9 11
这个问题其实并不难,重点在于什么时候更新花费。(PS:我比较喜欢用Dijkstra算法,它的原理这里就不详细说了,其实就是典型的贪心思想。)
在更新结点时,常规的最短路算法我们只考虑要加入的点是不是距离最小的,并没有考虑费用的问题。
那么有了费用,我们的问题是什么时候才会去更新费用呢?
分析:
如果说从起点start到node有多条路径都是最短的,那么就只存储保证最短路的同时,价值更小的那个路径经过的结点。
这样说可能不太明白,不过一看代码就显而易见了。
#include <iostream>#include <cstdio>#include <cstring>#define maxn 1005#define INF 0xffffffusing namespace std;int G[maxn][maxn];int V[maxn][maxn];int n,m;int Min(int x,int y){ return (x < y) ? x : y;}void init(){ for(int i = 0 ; i <= n ; ++i){ for(int j = 0 ; j <= n ; ++j){ G[i][j] = INF; V[i][j] = INF; } }}void Dijkstra(int s,int e){ int intree[maxn]; int mindist[maxn],minvalue[maxn]; int dist,value,pos; memset(intree,0,sizeof(intree)); for(int i = 1 ; i <= n ; ++i){ mindist[i] = G[s][i]; minvalue[i] = V[s][i]; } intree[s] = 1; for(int node = 1 ; node <= n-1 ; ++node){//剩余节点数 dist = INF; value = INF; for(int i = 1 ; i <= n ; ++i){ if(!intree[i]) if(mindist[i] < dist || (mindist[i] == dist && minvalue[i] < value)){//优先考虑路径最短 dist = mindist[i]; value = minvalue[i]; pos = i; } } intree[pos] = 1; //update for(int i = 1 ; i <= n ; ++i){ int tempdist = mindist[pos] + G[pos][i]; int tempvalue = minvalue[pos] + V[pos][i]; if(!intree[i]) if(tempdist < mindist[i] || (mindist[i] == tempdist && tempvalue < minvalue[i])){ mindist[i] = tempdist; minvalue[i] = tempvalue; } } } printf("%d %d\n",mindist[e],minvalue[e]);};int main(){ int v1,v2,v,len; int s,e; while(scanf("%d%d",&n,&m) != EOF){ if(n == 0 && m == 0) break; init(); while(m--){ scanf("%d%d%d%d",&v1,&v2,&len,&v); if(G[v1][v2] > len){ G[v1][v2] = G[v2][v1] = len; V[v1][v2] = V[v2][v1] = v; } } scanf("%d%d",&s,&e); Dijkstra(s,e); } return 0;}
0 0
- 杭电3790————最短路问题
- 杭电1217————不像最短路的"最短路"
- 最短路问题——Dijkstra算法
- 杭电3339————最短路(SPFA)+ 01背包
- Bellmen-Ford算法的应用——杭电OJ 2544 最短路
- 最短路—1001
- 最短路—1002
- 最短路—1003
- 最短路—1004
- 最短路—1005
- 最短路—1006
- 最短路—1007
- 最短路—1008
- 最短路—1009
- 百科—最短路
- 杭电 3790 最短路 Dijkstra算法
- 多段图的最短路问题——单向TSP问题
- 杭电 2544 最短路
- bmp_to_raw转BMP图片
- CSS学习历程《1》
- 嵌入式Linux 的虚拟文件系统剖析
- Eclipse中SVN的安装步骤(两种)和使用方法
- 第19章 主机名控制者:DNS服务器
- 杭电3790————最短路问题
- 汇编学习-分支与循环
- linux—select详解
- 【转】C# Mutex对象的使用
- 使用Fitnesse进行接口自动化测试
- You could try using --skip-broken to work around the problem
- js-闭包(closure)&this&arguments对象&懒加载(lazyload)
- iOS 判断当前设备第几次进入app及跳转app store
- Mongo客户端常用命令