HDU3790---(双权最短路径)

题目: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时,首先检查你有没有漏掉的情况。

 

经典:dijkstra

#include "cstdio"
#include "cstring"
#include "algorithm"
#define inf 0x3f3f3f3f
int dis[1002],vis[1002],cost[1002];///dis存储各点到点s的长度
typedef struct{///路径模型
    int lenth;///路长
    int cost;///路费
}Path;
Path map1[1002][1002];///地图
void dijkstra(int n,int s,int t)
{
    memset(vis,0,sizeof(vis));
    for(int i=1;i<=n;i++){
        dis[i]=map1[s][i].lenth;
        cost[i]=map1[s][i].cost;
    }

    int pos=1;
    dis[s]=0;
    vis[s]=1;
    for(int k=1;k<n;k++){
        int min1=inf;
        for(int i=1;i<=n;i++){
            if(!vis[i]&&min1>dis[i]){
                min1=dis[i];
                pos=i;
            }
        }
        vis[pos]=1;
        for(int i=1;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=1;i<=n;i++)
        {
           for(j=1;j<=n;j++)
            {
                map1[i][j].lenth=inf;
            }
        }

        int a,b,c,cost1;
        for(j=0;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 0;
}
时间: 2024-10-25 08:17:02

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

HDOJ 3790 最短路径问题 【双权值】

题意:... 难点:如何处理两个权值. 分析:题意说如果最短路径有多个,那么取价值最低的那个,所以说价值随着路径在变,如果路径不相等那么就更新路径并且更新价值,反之,则判断价值是不是要更新. 代码: #include<stdio.h> #include<string.h> #define M 1002 #define INF 0x3f3f3f3f int mapp[M][M], mapd[M][M], n, m, di[M], dp[M];//mapd是路径 mapp是价值 boo

HDOJ 3790 双权值Dijkstra

1 #include <iostream> 2 #include <stdio.h> 3 #include <string.h> 4 #include <cstring> 5 using namespace std; 6 7 const int INF = 1000000; 8 const int MAXSIZE = 1005; 9 10 int map[MAXSIZE][MAXSIZE]; 11 int price[MAXSIZE][MAXSIZE]; 1

HDOJ--3790--最短路径问题(双权值问题)

最短路径问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 18046    Accepted Submission(s): 5413 Problem Description 给你n个点,m条无向边,每条边都有长度d和花费p,给你起点s终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的. Input

HDU 3790-最短路径问题(双权最短路)

最短路径问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 13700    Accepted Submission(s): 4201 Problem Description 给你n个点,m条无向边,每条边都有长度d和花费p,给你起点s终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的. Input

杭电3790--最短路径问题(双权最小生成树)

最短路径问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 17425    Accepted Submission(s): 5199 Problem Description 给你n个点,m条无向边,每条边都有长度d和花费p,给你起点s终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的. Input

AOJ2249 最短路+最小花费(双权值)

写题解之前先骂一下这道题 xxx给数据范围点数<1e4,边数<2e4,结果我开2e4和3e4都RE,然后找问题一个多小时,最后我开了1e5和2e5,题面太能唬人了吧!?真是sb题面 ------------------------------------------分割线------------------------------------ 题目大意:给n个点和m条边,每条边给了起始点,距离和价格,求在保证点1到其他n-1个点的路径都是最短路的前提下所有路的价格之和的最小值 简单的Dijks

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

Dijkstra算法 1.定义概览 Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径.主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止.Dijkstra算法是很有代表性的最短路径算法, 在很多专业课程中都作为基本内容有详细的介绍,如数据结构,图论,运筹学等等.注意该算法要求图中不存在负权边. 问题描述:在无向图 G=(V,E) 中,假设每条边 E[i] 的长度为 w[i],找到由顶点 V0 到其余各点的最短路径.(单源最短路径) 2.算

浅入浅出数据结构(24)——最短路径问题

上一篇博文我们提到了图的最短路径问题:两个顶点间的最短路径该如何寻找?其实这个问题不应该叫"最短"路径问题,而应该叫"最便宜"路径问题,因为有时候我们会为图中的边赋权(weight),也叫权重,相当于经过一条边的"代价",一般为正数.比如下图(边旁的数字即该边的权重) 如果单纯考虑一条路径上边的条数,那么从v0到v6的最短路径应该是:v0-v3-v6.但是如果考虑边的权重,从v0到v6的"最便宜"路径应该是:v0-v1-v4-

最短路径算法整理

最短路径 1.概念 单源最短路径 单源最短路径实际是计算源点到其他各个顶点的最短路径的长度,常见算法有dijkstra算法 全局最短路径 全局最短路径实际是计算每个源点到其他各个顶点的最短路径的长度,我们可以调用dijkstra算法N次(这样没有Floyd算法快),常见解决全局最短路径的方法是Floyd-Warshall算法,但是Floyd-Warshall算法不能解决负边问题.为了解决负边问题,我们使用Bellman-Ford算法,或者SPFA(Shortest Path Faster Alg