PAT A1003 Emergency

PAT A1003 Emergency

题目简述:

原题为英文题目,所以在这里简述一下题意:

给定n个点和m条无向路以及起点、终点

下面一行n个数,第i个数表示第i个点上的救援组数目(点权

再往下m行每行三个整数,表示每一条路连接的两个端点以及花费(边权

要求求出从起点到终点的最短路径数目以及最短路径上的最大救援组数目和


解题思路:

这道题和一般的最短路稍有不同:本题除了寻找最短路还要额外维护两个值——最短路数目和救援组数目和最大值

  • 怎么处理呢?

增加两个维护类型,那我们也额外开两个数组记录呗:

一个数组记录最短路数目;一个数组记录完整的最短路路径,然后就可以根据路径来单独计算该条最短路径上的救援组数目,最后比较得出最大救援组数目和

  • 怎么维护最短路径数目?

我们使用ans[i]来存储第i号节点的最短路径数目

  1. 在跑Dijkstra时,如果dis[v]>dis[x]+e[i].val那么ans[v]=ans[x],因为不是最短路所以直接继承上一个节点的最短路数目即可

  2. 如果dis[v]==dis[x]+e[i].val那么ans[v]+=ans[x],因为最短路长度相等,所以累加最短路数目

  • 怎么记录一条完整的最短路路径?

记录一条完整路径,那我们记录在路径上每一个点的前驱即可

需要特别注意的是:

  1. 因为在本题中点的前驱不一定只有一个,所以我们需要使用vector(也许有其他做法,这里不多赘述)

  2. 当更新了当前的最短路时,当前节点v的前驱应该先清空,然后再把前一个点x存入前驱

来讲讲原因:

  1. 因为存在多条最短路径,而每条最短路径经过的点是可能重复的(这里还不涉及到救援组数目最大值),所以记录点的前驱需要使用vector

  2. 如果最短路都更新了,那么前面存的前驱也就没有用了,所以需要清空

if(dis[v]>dis[x]+e[i].val) {
ans[v]=ans[x];
pre[v].clear();
pre[v].push_back(x);
dis[v]=dis[x]+e[i].val;
shan.push(make_pair(-dis[v],v));
}
else if(dis[v]==dis[x]+e[i].val) {
ans[v]+=ans[x];
pre[v].push_back(x);
}
  • 记录了路径,那怎么维护救援组和的最大值?

我们在跑完Dijkstra部分后,进入dfs然后遍历vector来“复原”每一条最短路径

dfs中有两个参数:第一个now指当前的点,第二个sum表示至今的救援组数目和

如果now到达了起点,就用num比较出最大值

inline void dfs(int now,int sum) {
if(now==s) {
num=max(num,sum);
return ;
}
for(vector<int>::iterator it=pre[now].begin();it!=pre[now].end();it++) {
dfs(*it,sum+a[*it]);
}
}

完整Code:

解题思路如上所示,现在给出完整代码如下:

#include <bits/stdc++.h>
using namespace std;
int n,m,s,u,v,w,t,tot,num,a[250010];
int dis[250010],vis[250010],head[250010],ans[250010];
vector<int> pre[250010];
priority_queue<pair<int,int> > shan; struct node {
int to,net,val;
} e[250010]; inline void add(int u,int v,int w) {
e[++tot].val=w;
e[tot].to=v;
e[tot].net=head[u];
head[u]=tot;
} inline void dijkstra(int s) {
fill(dis,dis+250010,20050206);
dis[s]=0;
ans[s]=1; //起点肯定有路径,所以初始值为1
shan.push(make_pair(0,s));
while(!shan.empty()) {
int x=shan.top().second;
shan.pop();
if(vis[x]) continue;
vis[x]=1;
for(register int i=head[x];i;i=e[i].net) {
int v=e[i].to;
if(dis[v]>dis[x]+e[i].val) { //更新最短路
ans[v]=ans[x];
pre[v].clear(); //一定要清空前驱,再压入x
pre[v].push_back(x);
dis[v]=dis[x]+e[i].val;
shan.push(make_pair(-dis[v],v)); //入队
}
else if(dis[v]==dis[x]+e[i].val) { //如果最短路长度相等
ans[v]+=ans[x]; //累加最短路数目
pre[v].push_back(x); //同样记录前驱
}
}
}
} inline void dfs(int now,int sum) { //遍历所有最短路径求最大救援组数目和
if(now==s) { //回溯到起点,更新答案
num=max(num,sum);
return ;
}
for(vector<int>::iterator it=pre[now].begin();it!=pre[now].end();it++) { //利用vector不断遍历前驱
dfs(*it,sum+a[*it]);
}
} int main() {
scanf("%d%d%d%d",&n,&m,&s,&t);
for(register int i=0;i<n;i++) {
scanf("%d",&a[i]);
}
for(register int i=1;i<=m;i++) {
scanf("%d%d%d",&u,&v,&w);
add(u,v,w);
add(v,u,w);
}
dijkstra(s);
dfs(t,a[t]);
printf("%d %d",ans[t],num);
return 0;
}

最后,特别感谢一下RHL大佬对蒟蒻的指导qwq


PAT A1003 Emergency 题解的更多相关文章

  1. PAT甲题题解-1003. Emergency (25)-最短路径+路径数目

    给出n个城市,m条边,起始点c1和目的点c2接下来给出n个城市的队伍数以及m条双向边问你求c1到c2的所有最短路径数目,以及其中经过的最多队伍数 先最短路dijkstra,同时建立vector数组pr ...

  2. 2019秋季PAT甲级_C++题解

    2019 秋季 PAT (Advanced Level) C++题解 考试拿到了满分但受考场状态和知识水平所限可能方法不够简洁,此处保留记录,仍需多加学习.备考总结(笔记目录)在这里 7-1 Fore ...

  3. PAT 1003. Emergency (25)

    1003. Emergency (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yue As an emerg ...

  4. PAT 1003. Emergency (25) dij+增加点权数组和最短路径个数数组

    1003. Emergency (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yue As an emerg ...

  5. A1003. Emergency

    As an emergency rescue team leader of a city, you are given a special map of your country. The map s ...

  6. PAT 1003 Emergency

    1003 Emergency (25 分)   As an emergency rescue team leader of a city, you are given a special map of ...

  7. PAT甲题题解-1068. Find More Coins (30)-dp,01背包

    一开始没多想,虽然注意到数据N<=10^4的范围,想PAT的应该不会超时吧,就理所当然地用dfs做了,结果最后一组真的超时了.剪枝啥的还是过不了,就意识到肯定不是用dfs做了.直到看到别人说用0 ...

  8. PAT甲题题解-1108. Finding Average (20)-字符串处理

    求给出数的平均数,当然有些是不符合格式的,要输出该数不是合法的. 这里我写了函数来判断是否符合题目要求的数字,有点麻烦. #include <iostream> #include < ...

  9. PAT甲题题解-1128. N Queens Puzzle (20)-做了一个假的n皇后问题

    博主欢迎转载,但请给出本文链接,我尊重你,你尊重我,谢谢~http://www.cnblogs.com/chenxiwenruo/p/6789810.html特别不喜欢那些随便转载别人的原创文章又不给 ...

随机推荐

  1. Java实现 LeetCode 684 冗余连接(并查集)

    684. 冗余连接 在本问题中, 树指的是一个连通且无环的无向图. 输入一个图,该图由一个有着N个节点 (节点值不重复1, 2, -, N) 的树及一条附加的边构成.附加的边的两个顶点包含在1到N中间 ...

  2. Java实现 LeetCode 655 输出二叉树(DFS+二分)

    655. 输出二叉树 在一个 m*n 的二维字符串数组中输出二叉树,并遵守以下规则: 行数 m 应当等于给定二叉树的高度. 列数 n 应当总是奇数. 根节点的值(以字符串格式给出)应当放在可放置的第一 ...

  3. Java实现 蓝桥杯 算法训练 2的次幂表示

    算法训练 2的次幂表示 时间限制:1.0s 内存限制:512.0MB 问题描述 任何一个正整数都可以用2进制表示,例如:137的2进制表示为10001001. 将这种2进制表示写成2的次幂的和的形式, ...

  4. Java实现 LeetCode 315 计算右侧小于当前元素的个数

    315. 计算右侧小于当前元素的个数 给定一个整数数组 nums,按要求返回一个新数组 counts.数组 counts 有该性质: counts[i] 的值是 nums[i] 右侧小于 nums[i ...

  5. Java实现 蓝桥杯VIP 算法提高 最长公共子序列

    算法提高 最长公共子序列 时间限制:1.0s 内存限制:256.0MB 问题描述 给定两个字符串,寻找这两个字串之间的最长公共子序列. 输入格式 输入两行,分别包含一个字符串,仅含有小写字母. 输出格 ...

  6. java实现 蓝桥杯 算法训练 操作格子

    问题描述 有n个格子,从左到右放成一排,编号为1-n. 共有m次操作,有3种操作类型: 1.修改一个格子的权值, 2.求连续一段格子权值和, 3.求连续一段格子的最大值. 对于每个2.3操作输出你所求 ...

  7. Java实现无向图的欧拉回路判断问题

    1 问题描述 Problem Description 欧拉回路是指不令笔离开纸面,可画过图中每条边仅一次,且可以回到起点的一条回路.现给定一个图,问是否存在欧拉回路? Input 测试输入包含若干测试 ...

  8. Java实现三人年龄

    2 三人年龄 三个神秘蒙面人来访F博士. 博士询问他们年龄时,他们说:我们中年龄最小的不超过19岁.我们3人年龄总和为70岁.且我们三人年龄的乘积是所有可能情况中最大的. 请帮助F博士计算他们的年龄, ...

  9. Java实现第九届蓝桥杯乘积为零

    乘积为零 如下的10行数据,每行有10个整数,请你求出它们的乘积的末尾有多少个零? 5650 4542 3554 473 946 4114 3871 9073 90 4329 2758 7949 61 ...

  10. TZOJ 数据结构实验--静态顺序栈

    描述 创建一个顺序栈(静态),栈大小为5.能够完成栈的初始化.入栈.出栈.获取栈顶元素.销毁栈等操作. 顺序栈类型定义如下: typedef struct {  int data[Max];    i ...