洛谷 P8802 [蓝桥杯 2022 国 B] 出差

文章目录



[蓝桥杯 2022 国 B] 出差

题目链接

https://www.luogu.com.cn/problem/P8802

题目描述

A \mathrm{A} A 国有 N N N 个城市,编号为 1 ... N 1 \ldots N 1...N 小明是编号为 1 1 1 的城市中一家公司的员工,今天突然接到了上级通知需要去编号为 N N N 的城市出差。

由于疫情原因,很多直达的交通方式暂时关闭,小明无法乘坐飞机直接从城市 1 1 1 到达城市 N N N,需要通过其他城市进行陆路交通中转。小明通过交通信息网,查询到了 M M M 条城市之间仍然还开通的路线信息以及每一条路线需要花费的时间。

同样由于疫情原因,小明到达一个城市后需要隔离观察一段时间才能离开该城市前往其他城市。通过网络,小明也查询到了各个城市的隔离信息。(由于小明之前在城市 1 1 1,因此可以直接离开城市 1 1 1,不需要隔离)

由于上级要求,小明希望能够尽快赶到城市 N \mathrm{N} N, 因此他求助于你,希望你能帮他规划一条路线,能够在最短时间内到达城市 N N N 。

输入格式

第 1 1 1 行:两个正整数 N , M N, M N,M 表示 A 国的城市数量, M M M 表示末关闭的路线数量。

第 2 2 2 行: N N N 个正整数,第 i i i 个整数 C i C_{i} Ci 表示到达编号为 i \mathrm{i} i 的城市后需要隔离的时间。

第 3 ... M + 2 3 \ldots M+2 3...M+2 行: 每行 3 3 3 个正整数, u , v , c u, v, c u,v,c, 表示有一条城市 u u u 到城市 v v v 的双向路线仍然开通着,通过该路线的时间为 c c c。

输出格式

第 1 1 1 行: 1 1 1 个正整数,表示小明从城市 1 1 1 出发到达城市 N N N 的最短时间。(到达城市 N N N,不需要计算城市 N N N 的隔离时间)

样例 #1

样例输入 #1

4 4
5 7 3 4
1 2 4
1 3 5
2 4 3
3 4 5

样例输出 #1

13

提示

【样例说明】

【评测用例规模与约定】

对于 100 % 100 \% 100% 的数据, 1 ≤ N ≤ 1000 , 1 ≤ M ≤ 10000 , 1 ≤ C i ≤ 200 , 1 ≤ u , v ≤ 1 \leq N \leq 1000,1 \leq M \leq 10000,1 \leq C_{i} \leq 200,1 \leq u, v \leq 1≤N≤1000,1≤M≤10000,1≤Ci≤200,1≤u,v≤ N , 1 ≤ c ≤ 1000 N, 1 \leq c \leq 1000 N,1≤c≤1000

蓝桥杯 2022 国赛 B 组 E 题。



思路解析

求单源最短路问题,不过需要额外加上隔离天数的板子题。

最后别忘了减去城市 N N N 的隔离天数。


CODE

cpp 复制代码
#include <iostream>
#include <vector>
#include <cstring>
#include <algorithm>
#include <queue>
#define ll long long
#define INF 0x3f3f3f3f 

using namespace std;

typedef pair<int, int> pii;

const int N = 1010, M = 20010;
int h[N], e[M], ne[M], w[M], idx, c[N]; // 定义邻接表的数组,以及每个点的收费
int n, m; 		// 定义点数和边数
int dist[N]; 	// 定义到每个点的最短距离
bool st[N]; 	// 定义每个点是否在队列中的标记

void add(int a, int b, int c){
    e[idx] = b; 	// 存储边的终点
    ne[idx] = h[a]; // 存储边的下一条边的编号
    w[idx] = c; 	// 存储边的权值
    h[a] = idx++; 	// 更新头结点的编号
}

void spfa(){
    memset(dist, INF, sizeof dist); // 初始化距离为无穷大
    dist[1] = 0; 	// 起点到自己的距离为0
    
    queue<int> q; 	// 定义一个队列
    q.push(1); 		// 将起点入队
    st[1] = true; 	// 标记起点已经在队列中
    
    while(q.size()){ 	// 当队列不为空时循环
        int t = q.front(); // 取出队首元素
        q.pop(); 		// 将队首元素出队
        st[t] = false; 	// 标记该元素已经出队
        
        for(int i = h[t]; i != -1; i = ne[i]){ // 遍历该元素相邻的边
            int j = e[i]; // 获取边的终点
            
            // 如果可以用该边松弛终点的距离,即加上路线时间和隔离时间后比原来的距离小
            if(dist[j] > dist[t] + w[i] + c[j]){ 
                dist[j] = dist[t] + w[i] + c[j]; // 更新终点的距离
                
                if(!st[j]){ 	// 如果终点不在队列中
                    q.push(j); 	// 将终点入队
                    st[j] = true; // 标记终点已经在队列中
                }
            }
        }
    }
}

int main(){
    cin >> n >> m; 	// 输入城市数和路线数
    memset(h, -1, sizeof h); 	// 初始化邻接表头结点为-1
    
    for(int i = 1; i <= n; ++i)
        scanf("%d", &c[i]); 	// 输入每个城市的隔离时间
        
    while(m--){ 
        int a, b, c;
        scanf("%d%d%d", &a, &b, &c); 	// 输入一条路线的两个端点和时间
        add(a, b, c), add(b, a, c); 	// 将该路线加入邻接表,注意是双向路线,所以要加两次
    }
    
    spfa(); // 调用spfa算法求最短路
    
    // 输出到达城市n的最短时间,注意要减去城市n的隔离时间,因为题目要求不计算城市n的隔离时间
    cout << dist[n] - c[n] << endl; 
}

相关推荐
醒了就刷牙几秒前
Leetcode 面试150题 189. 轮转数组 中等
算法·leetcode·面试
<但凡.1 分钟前
力扣题解14——最长公共前缀
算法·leetcode
7yewh1 分钟前
LeetCode 力扣 热题 100道(十)回文链表(C++)
c语言·数据结构·c++·算法·leetcode·链表
JokerSZ.2 分钟前
【Leetcode 每日一题】235. 二叉搜索树的最近公共祖先
算法·leetcode·遍历
小咖拉眯3 分钟前
快速高效求素数|质数的方法—Java(模板)
java·开发语言·数据结构·算法
pursuit_csdn4 分钟前
LeetCode 2290. Minimum Obstacle Removal to Reach Corner
数据结构·算法·leetcode
pzx_0014 分钟前
【LeetCode】3208.交替组II
算法·leetcode·职场和发展
Adunn4 分钟前
算法基础 - 求解非线性方程(牛顿法)
开发语言·c++·算法
m0_675988234 分钟前
Leetcode3250:单调数组对的数目 I
数据结构·c++·算法·leetcode
茶猫_5 分钟前
力扣面试题 27 - 整数转换
算法·leetcode·职场和发展