题目:http://acm.hdu.edu.cn/showproblem.php?pid=3790

最短路径问题

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 25271    Accepted Submission(s): 7541

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)
 
Output
输出 一行有两个数, 最短距离及其花费。
 
Sample Input
3 2
1 2 5 6
2 3 4 5
1 3
0 0
 
Sample Output
9 11
 
方法:在经典dijkstra算法上稍加改动
 
分析:求最短路径的过程中,发现长度相等的两条路,选取花费少的。
   不过最大问题是、这题的最坑爹之处。认的测试数据里包含两个城市间有多条路径的情况
(所以两点间多条路径,无条件选择长度短的,长度相等,选择花费少的)
 
感悟:当你一直wrongAnswer时,首先检查你有没有漏掉的情况。
 
 
#include "cstdio"
#include "cstring"
#include "algorithm"
#define inf 0x3f3f3f3f
int dis[],vis[],cost[];///dis存储各点到点s的长度
typedef struct{///路径模型
int lenth;///路长
int cost;///路费
}Path;
Path map1[][];///地图
void dijkstra(int n,int s,int t)
{
memset(vis,,sizeof(vis));
for(int i=;i<=n;i++){///原点为S
dis[i]=map1[s][i].lenth;
cost[i]=map1[s][i].cost;
} int pos=;
dis[s]=;
vis[s]=;
for(int k=;k<n;k++){
int min1=inf;
for(int i=;i<=n;i++){
if(!vis[i]&&min1>dis[i]){
min1=dis[i];
pos=i;
}
}
vis[pos]=;
for(int i=;i<=n;i++){
int l=dis[pos]+map1[pos][i].lenth;
if(!vis[i]&&dis[i]>=l){
if(dis[i]==l){///找到相等路径时,选取花费少的
cost[i]=std::min(cost[i],cost[pos]+map1[pos][i].cost);
}
else///无条件选择路径短的
{
cost[i]=cost[pos]+map1[pos][i].cost;
dis[i]=l;
} }
}
}
}
int main()
{
int n,m,i,j;
Path p;
while(~scanf("%d%d",&n,&m)&&n&&m){
for(i=;i<=n;i++)
{
for(j=;j<=n;j++)
{
map1[i][j].lenth=inf;
}
} int a,b,c,cost1;
for(j=;j<m;j++)
{
scanf("%d%d%d%d",&a,&b,&c,&cost1);
///可能会出现重边!!!
if(c<map1[a][b].lenth){
map1[a][b].lenth=map1[b][a].lenth=c;///保留距离较短的
map1[a][b].cost=map1[b][a].cost=cost1;
}
else if(map1[a][b].lenth==c&&map1[a][b].cost>cost1)///保留费用较少的
{
map1[a][b].cost=map1[b][a].cost=cost1;
}
}
int s,t;
scanf("%d%d",&s,&t);
dijkstra(n,s,t);
printf("%d %d\n",dis[t],cost[t]);
}
return ;
}
 另一种

#include "cstdio"
#include<iostream>
#include<algorithm>
#include<queue>
#include<string.h>
#include<math.h>
#include<vector> using namespace std; struct node{
int ans = ;///动态记录起点到终点之间的最短花费
int minr = 1e9;///动态记录起点到终点之间的最短距离
vector<int>con;///另一点----数组包含所有与之连通的节点
vector<int>len;///之间长度----与con中点一一对应
vector<int>exp;///之间花费----与con中点一一对应
int vis = ;///节点是否已被访问
}data[]; int main()
{
int n, m, begi, endi;
while (cin >> n >> m)
{
if (n == && m == )
{
return ;
}
///初始化
for (size_t i = ; i <= n; i++)
{
data[i].ans = ;
data[i].vis = ;
data[i].minr = 1e9;
data[i].con.clear();
data[i].len.clear();
data[i].exp.clear();
}
///输入节点+权值信息 m条
for (size_t i = ; i < m; i++)
{
int be, ed, len, tar;
scanf("%d%d%d%d", &be, &ed, &len, &tar);
data[be].con.push_back(ed);
data[be].len.push_back(len);
data[be].exp.push_back(tar);
data[ed].con.push_back(be);
data[ed].len.push_back(len);///之间长度
data[ed].exp.push_back(tar);///之间花费
}
cin >> begi >> endi;
data[begi].ans = ;
data[begi].minr = ;
while ()
{
if (begi == endi)
{
break;
}
///查看所有与开始节点相连的节点
int size = data[begi].con.size();
for (size_t i = ; i < size; i++)///对剩余size-1个点
{
if (data[data[begi].con[i]].minr > data[begi].minr + data[begi].len[i])
{
data[data[begi].con[i]].ans = data[begi].ans + data[begi].exp[i];
data[data[begi].con[i]].minr = data[begi].minr + data[begi].len[i];
}
else if (data[data[begi].con[i]].minr == data[begi].minr + data[begi].len[i])
{
///路径路径相等,看花费
data[data[begi].con[i]].ans = min(data[begi].ans + data[begi].exp[i], data[data[begi].con[i]].ans);
}
}
data[begi].vis = ;
int temp = 1e9;
begi = -;
for (size_t i = ; i <= n; i++)
{
if (temp>data[i].minr&&data[i].vis == )///找出最小minr
{
temp = data[i].minr;
begi = i;
}
}
if (begi == -)
{
break;
}
}
cout << data[endi].minr << " " << data[endi].ans << "\n";
}
}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

HDU3790---(双权最短路径)的更多相关文章

  1. Dijkstra 算法——计算有权最短路径(边有权值)

    [0]README 0.1) 本文总结于 数据结构与算法分析, 源代码均为原创, 旨在理解 Dijkstra 的思想并用源代码加以实现: 0.2)最短路径算法的基础知识,参见 http://blog. ...

  2. 【迪杰斯特拉双关键字最短路】【HDU3790】【最短路径问题】

    题目大意: 给你n个点,m条无向边,每条边都有长度d和花费p,给你起点s终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的. 只需要再更新的时候判断一下就好 voi ...

  3. 【SPFA与Dijkstra的对比】CDOJ 1961 咸鱼睡觉觉【差分约束-负权最短路径SPFA】

    差分约束系统,求最小值,跑最长路. 转自:https://www.cnblogs.com/ehanla/p/9134012.html 题解:设sum[x]为前x个咕咕中至少需要赶走的咕咕数,则sum[ ...

  4. HDOJ 3790 双权值Dijkstra

    #include <iostream> #include <stdio.h> #include <string.h> #include <cstring> ...

  5. hdoj--3790--最短路径问题(双权值迪杰斯特拉)

     最短路径问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total ...

  6. 有向有权图的最短路径算法--Dijkstra算法

    Dijkstra算法 1.定义概览 Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径.主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止.Di ...

  7. PAT Advanced 1030 Travel Plan (30) [Dijkstra算法 + DFS,最短路径,边权]

    题目 A traveler's map gives the distances between cities along the highways, together with the cost of ...

  8. SLT 优先队列 哈弗曼树最小带权路径

    与普通的队列不同,普通的队列是先进先出的,而优先队列出队的顺序不是先进先出,而是大(或者小)元素先出队,需要#include <queue> 成员函数 成员函数 作用 empty() 判断 ...

  9. poj 题目分类(2)

    初期: 一.基本算法: (1)枚举. (poj1753,poj2965) (2)贪心(poj1328,poj2109,poj2586) (3)递归和分治法. (4)递推. (5)构造法.(poj329 ...

随机推荐

  1. 长沙Uber优步司机奖励政策(3月28日)

    滴快车单单2.5倍,注册地址:http://www.udache.com/ 如何注册Uber司机(全国版最新最详细注册流程)/月入2万/不用抢单:http://www.cnblogs.com/mfry ...

  2. shell -- 大括号和小括号

    在这里我想说的是几种shell里的小括号,大括号结构和有括号的变量,命令的用法,如下: 1.${var} 2.$(cmd) 3.()和{} 4.${var:-string},${var:+string ...

  3. Qt C++ 并发,并行,多线程编程系列1 什么是并发

    什么是并发,并发往简单来说就是两个或多个独立的任务同时发生,在我们的生活中也是随处可见.如果把每个人都当作一个独立的任务,那每个人可以相互独立的生活,这就是并发. 在计算机的系统里面,并发一般有两种, ...

  4. Qt Qwdget 汽车仪表知识点拆解4 另类进度条实现

    先贴上效果图,注意,没有写逻辑,都是乱动的 注意看一下,右面的这两个进度条,有瑕疵,就是我没有把图片处理干净,这里犹豫我不知道这个具体的弧度,也没法绘制,就偷懒了 现在上面放一个UI,把两个进度条抠空 ...

  5. jmeter完成CAS登录,并获取token(原创)

    思路: 1.系统完成CAS登录需要验证用户名/密码,以及动态授权参数 2.先通过指定url用正则提取出动态授权参数 3.完成登录需要cookie,需用正则提取出对应的cookie,已完成参数化的自动登 ...

  6. P4332三叉神经树

    题面 \(Solution\) 通过模拟,我们会发现每次修改 \(x\),只会改变从 \(x\) 向上一段连续的链的输出. 例如将 \(x\) 点从 \(0\) 改为 \(1,\) 那么它会影响从它向 ...

  7. 学习人工智能的第五个月[字典学习[Dictionary Learning,DL]]

    摘要: 大白话解释字典学习,分享第五个月的学习过程,人生感悟,最后是自问自答. 目录: 1.字典学习(Dictionary Learning,DL) 2.学习过程 3.自问自答 内容: 1.字典学习( ...

  8. 关闭电脑自带键盘(copy)

    用管理员身份运行cmd: 禁用笔记本键盘 sc config i8042prt start= disabled 启用笔记本键盘 sc config i8042prt start= auto

  9. LTE QCI分类 QoS

    http://blog.163.com/gzf_lte/blog/static/20840310620130140057204/ http://blog.163.com/gzf_lte/blog/st ...

  10. vue-component=>v-on

    $emit 返回 shouldPropagate,shouldPropagate 是一个布尔值,取决于父链上的是否存在该事件的监听器以及,事件处理程序返回的值.他决定 $dispatch 是否停止冒泡 ...