【蓝桥杯 2025 省 A 扫地机器人】题解

题目链接

如果 ∀ t i = 0 \forall ti = 0 ∀ti=0,直接输出 0 0 0。

由于题中所给是 n n n 个点 n n n 条边的 连通图,我们可以得到这是一棵基环树。

我们把答案分为三类:

  • 以环上某点 i i i 为根的树的直径;
  • 从某个树的叶子出发到达环上的根 i i i,绕过 整个环,回到 i i i,再走向 i i i 的另一个儿子直到叶子;
  • 从某个树的叶子出发到达环上的根 i i i,不绕过 整个环,走到环上另一点 j j j,再走向 j j j 的一个儿子直到叶子。

首先,我们找到基环树的环,环长为 m m m,然后对于每颗树,都预处理出 m d e p i \rm{mdepi} mdepi 和 s d e p i \rm{sdepi} sdepi,分别表示以 i i i 为根的树的深度最大值和次大值,这个一次 d f s \rm{dfs} dfs 就能做到。

然后对于长度为 m m m 的环,设顺时针(或逆时针)第 i i i 个点为 v e c i , i ∈ 0 , m ) \\rm{vec\[i}, \ i \in 0, m) vec\[i, i∈0,m),我们破环成链,令 v e c \[ i + m = v e c i \rm{veci + m = veci} veci+m=veci,再做一个前缀和,即

s i + 1 = s i + t v e c \[ i ] , i ∈ 0 , 2 m ) . \\rm{s\[i + 1 = si + tvec\[i]}, \ i \in 0, 2m). s\[i+1=si+tvec\[i], i∈[0,2m).

这样三类的答案就分别是:

  • 第一类:
    m d e p i + s d e p i + t i , ∀ i ∈ 0 , n ) . \\rm{mdep\[i + sdepi + ti}, \ \forall i \in 0, n). mdep\[i+sdepi+ti, ∀i∈[0,n).
  • 第二类:
    m d e p v e c \[ i ] + s d e p v e c \[ i ] + s m , ∀ i ∈ 0 , m ) . \\rm{mdep\[vec\[i] + sdepvec\[i] + sm, \ \forall i \in 0, m)}. mdep\[vec\[i]+sdepvec\[i]+sm, ∀i∈[0,m).
  • 第三类:
    m d e p v e c \[ i ] + ( s i + 1 − s j ) + m d e p v e c \[ j ] , 0 ≤ j < i < 2 m , i − j < m . \rm{mdepvec\[i] + (si + 1 - sj) + mdepvec\[j]}, \ 0 \leq j < i < 2m, \ i - j < m. mdepvec\[i]+(si+1−sj)+mdepvec\[j], 0≤j<i<2m, i−j<m.

第一、二类要求最大值直接 O ( n ) O(n) O(n) 遍历即可,第三类则需要对式子变形,变为

( m d e p v e c \[ i ] + s i + 1 ) + ( m d e p v e c \[ j ] − s j ) , \rm{(mdepvec\[i] + si + 1) + (mdepvec\[j] - sj)}, (mdepvec\[i]+si+1)+(mdepvec\[j]−sj),

这样一来对每个 i i i,就只要维护 m d e p v e c \[ j ] − s j \rm{mdepvec\[j] - sj} mdepvec\[j]−sj 的最大值,其中

j ∈ [ max ⁡ ( 0 , i − m + 1 ) , i ) , \rm{j \in [\max(0, i - m + 1), \ i),} j∈[max(0,i−m+1), i),

这可以用单调队列实现,时间复杂度 O ( n ) O(n) O(n)。

时间复杂度 O ( n ) O(n) O(n)

C++ Code

cpp 复制代码
#include <bits/stdc++.h>

template<class T>
std::istream &operator>>(std::istream &is, std::vector<T> &v) {
    for (auto &x: v) {
        is >> x;
    }
    return is;
}
template<class T>
void chmax(T &a, T b) {
    if (a < b) {
        a = b;
    }
}

int main() {
    std::ios::sync_with_stdio(false);
    std::cin.tie(nullptr);
    
    int n;
    std::cin >> n;

    std::vector<int> t(n);
    std::cin >> t;

    if (std::ranges::count(t, 0) == n) {
        std::cout << 0 << "\n";
        return 0;
    }

    std::vector<std::vector<int>> adj(n);
    for (int i = 0; i < n; i++) {
        int u, v;
        std::cin >> u >> v;
        u--, v--;
        adj[u].push_back(v);
        adj[v].push_back(u);
    }

    int idx = 0;
    std::vector<int> vec;
    vec.reserve(n);
    std::vector<bool> ins(n);
    std::vector<bool> cyc(n);
    std::vector<int> dfn(n, -1);
    std::vector<int> pre(n, -1);
    auto init = [&](auto &&self, int x) -> void {
        dfn[x] = idx++;
        ins[x] = true;
        for (int y: adj[x]) {
            if (y == pre[x]) {
                continue;
            }
            if (dfn[y] == -1) {
                pre[y] = x;
                self(self, y);
            } else if (ins[y]) {
                for (int i = x; ; i = pre[i]) {
                    vec.push_back(i);
                    cyc[i] = true;
                    if (i == y) {
                        break;
                    }
                }
            }
        }
        ins[x] = false;
    };
    init(init, 0);

    std::vector<int> mdep(n);
    std::vector<int> sdep(n);
    std::ranges::fill(pre, -1);
    auto tree = [&](auto &&self, int x) -> void {
        for (int y: adj[x]) {
            if (y == pre[x] or cyc[y]) {
                continue;
            }
            pre[y] = x;
            self(self, y);
            if (mdep[x] < mdep[y] + t[y]) {
                sdep[x] = mdep[x];
                mdep[x] = mdep[y] + t[y];
            } else if (sdep[x] < mdep[y] + t[y]) {
                sdep[x] = mdep[y] + t[y];
            }
        }
    };

    int res = 0;
    for (int i: vec) {
        tree(tree, i);
        chmax(res, mdep[i] + sdep[i]);
    }
    int dia = 0;
    for (int i = 0; i < n; i++) {
        chmax(dia, mdep[i] + sdep[i] + t[i]);
    }
    
    int m = vec.size();
    vec.insert(vec.end(), vec.begin(), vec.end());

    std::vector<int> s(2 * m + 1);
    for (int i = 0; i < 2 * m; i++) {
        s[i + 1] = s[i] + t[vec[i]];
    }
    int ans = std::max(dia, res + s[m]);

    std::deque<int> q;
    for (int i = 0; i < 2 * m; i++) {
        int j = vec[i];
        while (not q.empty() and i - q.front() >= m) {
            q.pop_front();
        }
        if (not q.empty()) {
            chmax(ans, mdep[j] + s[i + 1] + mdep[vec[q.front()]] - s[q.front()]);
        }
        while (not q.empty() and mdep[vec[q.back()]] - s[q.back()] <= mdep[j] - s[i]) {
            q.pop_back();
        }
        q.push_back(i);
    }

    std::cout << ans << "\n";

    return 0;
}
相关推荐
JieE21216 小时前
LeetCode 101. 对称二叉树|JS 递归 + 迭代双解法,彻底搞懂镜像判断
javascript·算法
JieE2122 天前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack202 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树2 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE2123 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2123 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术3 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦3 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
clint4563 天前
C++进阶(1)——前景提要
c++