第十四届蓝桥杯省赛C++B组I题【景区导游】题解(AC)

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

解题思路

题目已给出地图为一个 n n n 个点, n − 1 n-1 n1 条路线的树。

对于计算树中任意两点的距离,我们可以使用 LCA 算法进行快速计算。

假设 a , b a, b a,b 的最近公共祖先为 c c c,那么 a , b a,b a,b 之间的距离为 d i s t [ a ] + d i s t [ b ] − 2 × d i s t [ c ] dist[a]+dist[b]-2\times dist[c] dist[a]+dist[b]2×dist[c],其中 d i s t [ x ] dist[x] dist[x] 表示从根节点到 x x x 点的距离。

所以我们可以先算出原定路线所消耗的时间 r e s res res,再进行删点删边操作:

  1. 假设删掉的为第 1 1 1 个点,相当于删掉从 1 1 1 号点到 2 2 2 号点的边,相当于从 2 2 2 号点开始。那么答案应该等于
    r e s − g e t _ d i s t ( q [ 1 ] , q [ 2 ] ) res - get\_dist(q[1], q[2]) resget_dist(q[1],q[2])
  2. 假设删掉的为第 i ( 1 < i < m ) i(1<i<m) i(1<i<m) 个点,相当于删掉从 i − 1 i-1 i1 号点到 i i i 号点的边和从 i i i 号点到 i + 1 i+1 i+1 号点的边,并且补上一条从 i − 1 i-1 i1 号点到 i + 1 i+1 i+1 号点的边,那么答案应该等于
    r e s − g e t _ d i s t ( q [ i − 1 ] , q [ i ] ) − g e t _ d i s t ( q [ i ] , q [ i + 1 ] ) + g e t _ d i s t ( q [ i − 1 ] , q [ i + 1 ] ) res - get\_dist(q[i - 1], q[i]) - get\_dist(q[i], q[i + 1]) + get\_dist(q[i - 1], q[i + 1]) resget_dist(q[i1],q[i])get_dist(q[i],q[i+1])+get_dist(q[i1],q[i+1])
  3. 假设删掉的为第 m m m 个点,相当于删掉从 m − 1 m-1 m1 号点到 m m m 号点的边,相当于从 m − 1 m-1 m1 号点结束。那么答案应该等于
    r e s − g e t _ d i s t ( q [ m − 1 ] , q [ m ] ) res - get\_dist(q[m-1], q[m]) resget_dist(q[m1],q[m])

AC_Code

#include <iostream>
#include <cstring>
#include <algorithm>
#include <queue>

using namespace std;

typedef long long LL;

const int N = 1e5 + 10, M = N * 2;

int n, m;
int h[N], e[M], w[M], ne[M], idx;
int depth[N];
LL dist[N];
int fa[N][20];
int q[N];

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

void bfs()
{
    memset(depth, 0x3f, sizeof depth);
    depth[0] = 0, depth[1] = 1;
    queue<int> q;
    q.push(1);

    while (q.size())
    {
        auto t = q.front();
        q.pop();

        for (int i = h[t]; ~i; i = ne[i] )
        {
            int j = e[i];
            if (depth[j] <= depth[t] + 1)
                continue;
            depth[j] = depth[t] + 1;
            dist[j] = dist[t] + w[i];
            fa[j][0] = t;
            for (int k = 1; k <= 19; ++ k )
                fa[j][k] = fa[fa[j][k - 1]][k - 1];
            q.push(j);
        }
    }
}

int get_lca(int a, int b)
{
    if (depth[a] < depth[b])
        swap(a, b);
    for (int i = 19; i >= 0; -- i )
        if (depth[fa[a][i]] >= depth[b])
            a = fa[a][i];
    if (a == b)
        return a;

    for (int i = 19; i >= 0; -- i )
        if (fa[a][i] != fa[b][i])
        {
            a = fa[a][i];
            b = fa[b][i];
        }

    return fa[a][0];
}

LL get_dist(int a, int b)
{
    int c = get_lca(a, b);
    return dist[a] + dist[b] - dist[c] * 2;
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0);
    cout.tie(0);

    cin >> n >> m;

    memset(h, -1, sizeof h);

    for (int i = 0; i < n - 1; ++ i )
    {
        int a, b, c;
        cin >> a >> b >> c;
        add(a, b, c), add(b, a, c);
    }

    for (int i = 1; i <= m; ++ i )
        cin >> q[i];

    bfs();

    LL res = 0;
    for (int i = 1; i <= m - 1; ++ i )
        res += get_dist(q[i], q[i + 1]);

    for (int i = 1; i <= m; ++ i )
    {
        if (i == 1)
            cout << res - get_dist(q[1], q[2]) << ' ';
        else if (i < m)
            cout << res - get_dist(q[i - 1], q[i]) - get_dist(q[i], q[i + 1]) + get_dist(q[i - 1], q[i + 1]) << ' ';
        else
            cout << res - get_dist(q[m - 1], q[m]) << ' ';
    }

    cout << endl;

    return 0;
}

【在线测评】

在这里插入图片描述

相关推荐

  1. C++B题解

    2024-07-19 17:12:06       39 阅读
  2. 2023C/C++大学A题解

    2024-07-19 17:12:06       33 阅读
  3. Python大学BI题解

    2024-07-19 17:12:06       24 阅读
  4. Python(未完)

    2024-07-19 17:12:06       36 阅读
  5. 大学B填空(c++)

    2024-07-19 17:12:06       35 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-07-19 17:12:06       70 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-07-19 17:12:06       74 阅读
  3. 在Django里面运行非项目文件

    2024-07-19 17:12:06       62 阅读
  4. Python语言-面向对象

    2024-07-19 17:12:06       72 阅读

热门阅读

  1. 【C++】C++中find_first_of函数解析

    2024-07-19 17:12:06       20 阅读
  2. PHP MySQL 读取数据

    2024-07-19 17:12:06       19 阅读
  3. Handler续谈(epoll)

    2024-07-19 17:12:06       19 阅读
  4. Git提交到错误分支怎么办?(解决办法)

    2024-07-19 17:12:06       22 阅读
  5. 在ubuntu系统上安装qt 2

    2024-07-19 17:12:06       20 阅读
  6. CONFIG_MTD_SPI_NOR_USE_4K_SECTORS

    2024-07-19 17:12:06       23 阅读
  7. 网络通信协议

    2024-07-19 17:12:06       14 阅读
  8. opencv 使用XML和YAML格式来输入输出文件

    2024-07-19 17:12:06       26 阅读
  9. css2024

    2024-07-19 17:12:06       21 阅读