第十四届蓝桥杯大赛软件赛省赛 C/C++ 大学 A 组 G题

问题描述

在这里插入图片描述
在这里插入图片描述


格式输入

输入的第一行包含三个整数 n, m, q ,分别表示设备数、物理连接数和询问数。接下来 m 行,每行包含三个整数 ui, vi,wi ,分别表示 ui 和 vi 之间有一条稳定性为 wi 的物理连接。
接下来 q 行,每行包含两个整数 xi, yi ,表示查询 xi 和 yi 之间的通信稳定性。


格式输出

输出 q 行,每行包含一个整数依次表示每个询问的答案。


样例输入

5 4 3
1 2 5
2 3 6
3 4 1
1 4 3
1 5
2 4
1 3


样例输出

-1
3
5


评测用例规模与约定

对于 30% 的评测用例,n, q ≤ 500,m ≤ 1000 ;
对于 60% 的评测用例,n, q ≤ 5000,m ≤ 10000 ;
对于所有评测用例,2 ≤ n, q ≤ 10^5,1 ≤ m ≤ 3 × 10^5,1 ≤ ui, vi, xi, yi ≤ n,
1 ≤ wi ≤ 10^6,ui!= vi,xi!=yi 。


解析

由于本题中所有边权值均为正数,因此可以使用 Dijkstra 算法进行优化。从起点开始,依次扫描与其相连的所有边,更新到各个顶点的最小距离信息。对于每一个询问,只需使用 Dijkstra 算法计算出从 xi 到 yi 的所有可行路径,然后取这些路径中的最小边权值,即为 xi 和 yi 之间通信的稳定性。当然,如果不存在路径,则输出 -1。


参考程序

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

const int N = 10010, M = 2 * N;
typedef pair<int, int> PII;
int h[N], e[M], ne[M], w[M], idx;
int dist[N];
bool st[N];
int n, m, q;

void add(int a, int b, int c)
{
    e[idx] = b, w[idx] = c, ne[idx] = h[a], h[a] = idx++;
}

void dijkstra(int start)
{
    memset(dist, 0x3f, sizeof dist);
    memset(st, false, sizeof st);
    priority_queue<PII, vector<PII>, greater<PII>> heap;
    heap.push({ 0, start });
    dist[start] = 0;

    while (heap.size())
    {
        auto t = heap.top();
        heap.pop();

        int ver = t.second, distance = t.first;
        if (st[ver]) continue;
        st[ver] = true;

        for (int i = h[ver]; ~i; i = ne[i])
        {
            int j = e[i];
            if (dist[j] > max(dist[ver], w[i]))
            {
                dist[j] = max(dist[ver], w[i]);
                heap.push({ dist[j], j });
            }
        }
    }
}

int main()
{
    scanf("%d%d%d", &n, &m, &q);
    memset(h, -1, sizeof h);

    while (m--)
    {
        int a, b, c;
        scanf("%d%d%d", &a, &b, &c);
        add(a, b, c), add(b, a, c);
    }

    while (q--)
    {
        int a, b;
        scanf("%d%d", &a, &b);
        dijkstra(a);
        if (dist[b] == 0x3f3f3f3f) puts("-1");
        else printf("%d\n", dist[b]);
    }

    return 0;
}

最大生成树+lca

#include <stdio.h>
#include <algorithm>

const int N = 100005;

struct edge {
    int u, v, w;
    inline bool operator<(edge e) { return w > e.w; };
} edges[3 * N];

int head[N << 1], query[N], next[N << 2], ver[N << 2], wei[N << 1];

int n, m, p, q, cur, ans[N], linked[N << 1], w[N << 1], buffer[N << 1];

int find(int x) { return x == linked[x] ? x : (linked[x] = find(linked[x])); }

bool vis[N << 1];

void dfs(int u) {
    linked[u] = u;
    buffer[p++] = u;
    vis[u] = 1;
    for (int i = head[u]; i; i = next[i])
        dfs(ver[i]), linked[ver[i]] = u;
    for (int i = query[u]; i; i = next[i])
        if (vis[ver[i]]) ans[wei[i]] = w[find(ver[i])];
}

int main() {
    scanf("%d %d %d", &n, &m, &q);
    for (int i = 0; i < m; ++i)
        scanf("%d %d %d", &edges[i].u, &edges[i].v, &edges[i].w);
    for (int i = 0, x, y; i < q; ++i) {
        scanf("%d %d", &x, &y);
        next[++cur] = query[x], query[x] = cur, ver[cur] = y, wei[cur] = i;
        next[++cur] = query[y], query[y] = cur, ver[cur] = x, wei[cur] = i;
    }
    std::sort(edges, edges + m);
    for (int i = (n << 1) - 1; i; --i) linked[i] = i;
    for (int i = 0, x = n; i < m; ++i) {
        int u = find(edges[i].u), v = find(edges[i].v);
        if (u == v) continue;
        w[++x] = edges[i].w;
        linked[u] = linked[v] = x;
        next[++cur] = head[x], head[x] = cur, ver[cur] = u;
        next[++cur] = head[x], head[x] = cur, ver[cur] = v;
    }
    for (int i = (n << 1) - 1; i; --i, p = 0) {
        if (linked[i] == i) dfs(i);
        while (p--) vis[buffer[p]] = 0;
    }
    for (int i = 0; i < q; ++i)
        printf("%d\n", ans[i] ? ans[i] : -1);
}

以个人刷题整理为目的,如若侵权,请联系删除~

  • 6
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 8
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值