题目: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. 在WPF中自定义控件(3) CustomControl (下)

    原文:在WPF中自定义控件(3) CustomControl (下)   在WPF中自定义控件(3) CustomControl (下)                                 ...

  2. Django学习之天气调查实例(2):显示数据表数据

    数据表数据添加后,如添加3条用户信息,分别为“aaa”.“bbb”.“ccc”,现在通过代码的方式显示数据表中的数据. 1.在website项目文件夹中创建 userload.py文件,并且写如下代码 ...

  3. Hystrix入门指南

    Introduction 1.Where does the name come from? hystrix对应的中文名字是“豪猪”,豪猪周身长满了刺,能保护自己不受天敌的伤害,代表了一种防御机制,这与 ...

  4. create subnet

    子网相关功能点: 模块 功能 描述 备注 子网 创建子网 创建一个子网   设置子网网段范围   设置子网网关IP/不开启网关   给子网开启/关闭dhcp   设置子网dns   修改子网 修改子网 ...

  5. 1.16. BIP39协议:使用助记词生成确定性钱包

    以太坊系统学习教程: https://www.netkiller.cn/blockchain/bip39.html 1.16. BIP39协议:使用助记词生成确定性钱包 BIP:39 层:应用层 标题 ...

  6. C++STL——队列

    一.相关定义 原理:queue 队列也是一个线性存储表,元素数据的插入在表的一端进行,在另一端删除,从而构成了一个先进先出FIFO(First In First Out)表. 队头&队尾:插入 ...

  7. linux消息队列通信

    IPC机制 进程间通信机制(Inter Process Communication,IPC),这些IPC机制的存在使UNIX在进程通信领域手段相当丰富,也使得程序员在开发一个由多个进程协作的任务组成的 ...

  8. const用法详解(转)

    http://www.cnblogs.com/StudyRush/archive/2010/10/06/1844690.html 面向对象是C++的重要特性. 但是c++在c的基础上新增加的几点优化也 ...

  9. 不错的PDF开发库

    C++库: 1,PDF类库 PoDoFo   http://podofo.sourceforge.net/  PoDoFo 是一个用来操作 PDF 文件格式的 C++ 类库.它还包含一些小工具用来解析 ...

  10. 正则表达式之旅_sed_awk

    谈谈正则表达式这个东西: 我想作为一个程序员,正则表达式大家绝对不陌生. 正则表达式好像一个有限则动机.主要作用是匹配,但是同时因为这个功能,我们可以扩展很多其他用法 像很多语言都引人了正则表达式:j ...