L2-001 城市间紧急救援 (25分)

作为一个城市的应急救援队伍的负责人,你有一张特殊的全国地图。在地图上显示有多个分散的城市和一些连接城市的快速道路。每个城市的救援队数量和每一条连接两个城市的快速道路长度都标在地图上。当其他城市有紧急求助电话给你的时候,你的任务是带领你的救援队尽快赶往事发地,同时,一路上召集尽可能多的救援队。

输入格式:

输入第一行给出4个正整数N、M、S、D,其中N(2≤N≤500)是城市的个数,顺便假设城市的编号为0 ~ (N−1);M是快速道路的条数;S是出发地的城市编号;D是目的地的城市编号。
第二行给出N个正整数,其中第i个数是第i个城市的救援队的数目,数字间以空格分隔。随后的M行中,每行给出一条快速道路的信息,分别是:城市1、城市2、快速道路的长度,中间用空格分开,数字均为整数且不超过500。输入保证救援可行且最优解唯一。

输出格式:

第一行输出最短路径的条数和能够召集的最多的救援队数量。第二行输出从S到D的路径中经过的城市编号。数字间以空格分隔,输出结尾不能有多余空格。

输入样例:

4 5 0 3
20 30 40 10
0 1 1
1 3 2
0 3 3
0 2 2
2 3 2

输出样例:

2 60
0 1 3

其实就是dijkstra算法,固定套路+根据题意添加变量做修改。
只是数据量稍微多些,我把数据量分为了三类。
1.输入 2. 输出3. 固定套路数据量
记住算法中固定套路的数据量比如d[maxn] ,vis[maxn],w[maxn]等再根据题意添加数据量这样会简单些。
具体思路:
首先分析题目明确目的,要求输出最短路径条数,还要能够召集最多的救援队数量,还有输出经过的城市编号。

接下来开始求解。从起点,找距离最近的城市点u,找到后标记u已被访问,确定u作为中转站,通过d[v] > d[u] + G[u][v]更新dv;反复重复n次,直到所有城市点都被标记。这就是一般dijskra大致思路再结合目的细化考虑。
最短路径条数,召集最多救援队数量,记录前结点都应该伴随在更新dv的步骤后。更新dv条件:d[v] <d[u] + G[u][v] || (d[v] == d[u] + G[u][v] && w[v] < w[u] + weight[v]

最后,深度遍历输出
注意: 要记得d[maxn] 、G[maxn][maxn]初始化为INF

#include <iostream>
#include <cstdio>
#include <memory>
#include <algorithm>
using namespace std;
const int maxn = 550;
const int INF = 0x3fffffff;
int n,m,s,e;
int G[maxn][maxn], weight[maxn];
int w[maxn],d[maxn],pre[maxn],num[maxn];
bool visit[maxn] = {false};

void dfs(int v)
{
    if(v == s) 
    {
        cout << s;
        return;
    }
    else
    {
        dfs(pre[v]);
        cout << " " << v;
    }
}
void dijkstra(int s)
{
    fill(d, d + maxn, INF);
    fill(w, w + maxn, 0);
    fill(num, num + maxn, 0);
    d[s] = 0;
    w[s] = weight[s];
    num[s] = 1;
    for(int i = 0; i < n; i++)
    {
        int u = -1, MIN = INF; 
        for(int j = 0; j < n; j++)
        {
            if(visit[j] == false && d[j] < MIN)
            {
                u = j;
                MIN = d[u];
            }
        }
        if(u == -1) return;
        visit[u] = true;
        for(int v = 0; v < n; v++)
        {
            if(visit[v] == false && G[u][v] != INF)
            {
                if(d[u] + G[u][v] < d[v])
                {
                    d[v] = d[u] + G[u][v];
                    w[v] = w[u] + weight[v];
                    num[v] = num[u];
                    pre[v] = u;
                }
                else if(d[u] + G[u][v] == d[v])
                    {
                        if(w[v] < w[u] + weight[v])
                        {
                            w[v] = w[u] + weight[v];
                            pre[v] = u;
                        }
                        num[v] += num[u];
                    }
                }
            }
        }
}
int main()
{
    fill(G[0],G[0]+maxn * maxn, INF); //都是不可达的
    cin >> n >> m >> s >> e;
    for(int i = 0; i < n; i++)
    {
        cin >> weight[i];
    }
    while(m--)
    {
        int u,v;
        cin >> u >> v;
        cin >> G[u][v];
        G[v][u] = G[u][v];
    }
    dijkstra(s);
    cout << num[e] << " " << w[e] << endl;
    dfs(e);
    return 0;
}
  • 4
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

moumoumouwang

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值