2022年第十三届蓝桥杯大赛软件类决赛C/C++大学B组E题出差

本文介绍了一种使用Dijkstra算法求解特殊条件下的最短路径问题的方法,即考虑到城市间的旅行需经过中转及隔离的情况。通过调整标准Dijkstra算法,在每次松弛操作时增加隔离时间来解决这个问题。

题意:

有N个城市,编号1...N,无法从1出发到N,需要通过其他城市中转,并且到达后需隔离,求1到N时间最短的路线。

思路:

最短路变形,求时间最短,使用Dijkstra算法松弛时需加上隔离的时间,注意,到达城市N不用隔离。

代码:

#include<iostream>
#include<cstring>
#include<algorithm>
#include<vector>
#include<queue>
using namespace std;

const int N = 1010;
const int INF = 0x3f3f3f3f;
int n, m, t[N], d[N];
bool vis[N];
struct Edge {
    int u, v, w;
    Edge(int u, int v, int w): u(u), v(v), w(w) {}
};
struct Node {
    int u, d;
    bool operator < (const Node& a) const {
        return d > a.d;
    }
    Node(int u, int d): u(u), d(d) {}
};
vector<Edge> edges;
vector<int> G[N];
int dijkstra() {
    priority_queue<Node> q;
    q.push(Node(1, 0));
    memset(d, INF, sizeof(d));
    memset(vis, 0, sizeof(vis));
    d[1] = 0;
    while(!q.empty()) {
        Node x = q.top(); q.pop();
        if(vis[n])
            break;
        if(vis[x.u])
            continue;
        vis[x.u] = true;
        for(int i=0; i<G[x.u].size(); i++) {
            auto& e = edges[G[x.u][i]];
            if(e.v == n) {
                if(d[x.u] + e.w < d[e.v]) {//到达城市N不需隔离,也可输出时直接减去隔离时间
                    d[e.v] = d[x.u] + e.w;
                    q.push(Node(e.v, d[e.v]));
                }

            }
            else {
                if(d[x.u] + e.w + t[e.v] < d[e.v]) {
                    d[e.v] = d[x.u] + e.w + t[e.v];
                    q.push(Node(e.v, d[e.v]));
                }
            }
        }
    }
    return d[n];
}

int main() {
    cin>>n>>m;
    for(int i=1; i<=n; i++)
        cin>>t[i];
    int u, v, w;
    for(int i=0; i<m; i++) {
        cin>>u>>v>>w;
        edges.push_back(Edge(u, v, w));
        edges.push_back(Edge(v, u, w));//双向路线
        G[u].push_back(i*2);
        G[v].push_back(i*2+1);
    }
    int ans = dijkstra();
    cout<<ans<<endl;
}

在C语言里,位左对齐右对齐一般在格式化输出时会用到,主要用于控制数据在输出时的位置。以下是相关介绍: ### 整型数据的左对齐右对齐 通过`printf`函数实现整型数据的左对齐右对齐右对齐是默认方式,在格式说明符`%`和`d`之间添加数字来规定输出宽度,若数字位数小于规定宽度,会在左边补空格;左对齐则需在数字前加`-`号,若数字位数小于规定宽度,会在右边补空格。 示例代码如下: ```c #include <stdio.h> int main() { // 右对齐。数字宽度为10,若不足10,在左边补足空格 printf("%10d\n", 1234); // 左对齐。数字宽度为10,若不足10,在右边补足空格 printf("%-10d\n", 1234); return 0; } ``` ### 不同输出长度的情况 当规定的输出宽度和数字实际位数不同时,有不同的处理方式。若规定宽度小于数字实际位数,会完整输出数字;若规定宽度大于数字实际位数,右对齐在左边补空格,左对齐在右边补空格。 示例代码如下: ```c #include <stdio.h> int main() { // -5是左对齐,输出长度为5。5是右对齐,输出长度为5 printf("%-5d %5d\n", 455, 455); printf("%-5d %5d\n", -123, -123); // 规定宽度小于实际位数,完整输出数字 printf("%-5d %5d\n", 987654, 987654); return 0; } ``` ### 其他数据类型的对齐 除整型外,其他数据类型也能实现左对齐右对齐。例如浮点数(`%f`)、字符串(`%s`)等,方法和整型一致。 示例代码如下: ```c #include <stdio.h> int main() { // 右对齐浮点数,宽度为10 printf("%10f\n", 3.14); // 左对齐浮点数,宽度为10 printf("%-10f\n", 3.14); // 右对齐字符串,宽度为10 printf("%10s\n", "hello"); // 左对齐字符串,宽度为10 printf("%-10s\n", "hello"); return 0; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

我恨TLE

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

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

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

打赏作者

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

抵扣说明:

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

余额充值