PAT L2-001 紧急救援

https://pintia.cn/problem-sets/994805046380707840/problems/994805073643683840

 

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

输入格式:

输入第一行给出4个正整数N、M、S、D,其中N(2)是城市的个数,顺便假设城市的编号为0 ~ (;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

代码:

#include <bits/stdc++.h>
using namespace std;

const int inf = 0x3f3f3f3f;
int mp[550][550];
int N, M, S, D;
int dis[550], vis[550], team[550], pre[550], ha[550], num[550];

void dijkstra(int act) {
    dis[act] = 0;
    pre[act] = -1;
    ha[act] = team[act];
    num[act] = 1;

    for(int i = 0; i < N; i ++) {
        int minn = inf;
        int temp = -1;
        for(int j = 0; j < N; j ++) {
            if(vis[j]) continue;
            if(temp == -1) temp = j;
            if(dis[j] < minn || dis[j] == minn && ha[j] > ha[temp]) {
                minn = dis[j];
                temp = j;
            }
        }

        if(temp == -1) break;
        vis[temp] = 1;
        for(int k = 0; k < N; k ++) {
            if(vis[k]) continue;
            if(mp[temp][k] == inf) continue;

            if(dis[k] > dis[temp] + mp[temp][k]) {
                dis[k] = dis[temp] + mp[temp][k];
                ha[k] = ha[temp] + team[k];
                pre[k] = temp;
                num[k] = num[temp];
            } else if(dis[k] == dis[temp] + mp[temp][k]) {
                num[k] += num[temp];

                if(ha[temp] + team[k] > ha[k]) {
                    ha[k] = ha[temp] + team[k];
                    pre[k] = temp;
                }
             }
        }
    }
}

int main() {
    scanf("%d%d%d%d", &N, &M, &S, &D);
    for(int i = 0; i < N; i ++)
        scanf("%d", &team[i]);

    memset(mp, inf, sizeof(mp));
    memset(dis, inf, sizeof(dis));
    memset(vis, 0, sizeof(vis));
    for(int i = 0; i < M; i ++) {
        int st, en, val;
        scanf("%d%d%d", &st, &en, &val);
        mp[st][en] = mp[en][st] = val;
    }

    dijkstra(S);

    printf("%d %d\n", num[D], ha[D]);

    stack<int> s;
    int last = D;
    while(last != -1) {
        s.push(last);
        last = pre[last];
    }

    printf("%d", s.top());
    s.pop();
    while(!s.empty()) {
        printf(" %d", s.top());
        s.pop();
    }
    return 0;
}

  $O(N^2)$

被学分支配的恐惧 算来算去算不清 选课是最让 Zlrrr 头大的事情了

posted @ 2019-03-06 11:53  丧心病狂工科女  阅读(288)  评论(0编辑  收藏  举报