加权重心(换根DP)

题目

换根DP,手推一下即可。

代码如下:

cpp 复制代码
#include <bits/stdc++.h>
#define int long long
#define MAXN 100010
using namespace std;

int N, A, B, L, C, minn = 0x3f3f3f3f3f3f3f3f, size[MAXN], dp[MAXN];
vector<pair<int, int>> tree[MAXN];

void dfs1(int u, int pa) {
    size[u] = tree[u][0].first;
    for (int i = 1; i < (int) tree[u].size(); i++) {
        int v = tree[u][i].first, w = tree[u][i].second;
        if (v == pa)
            continue;
        dfs1(v, u);
        size[u] += size[v];
        dp[u] += dp[v] + size[v] * w;
    }
}

void dfs(int u, int pa) {
    for (int i = 1; i < (int) tree[u].size(); i++) {
        int v = tree[u][i].first, w = tree[u][i].second;
        if (v == pa)
            continue;
        dp[v] = dp[u] - size[v] * w + (size[1] - size[v]) * w;
        dfs(v, u);
    }
}

signed main () {
    scanf("%lld", &N);
    for (int i = 1; i <= N; i++) {
        scanf("%lld", &C);
        tree[i].push_back({C, 0});
    }
    for (int i = 1; i < N; i ++) {
        scanf("%lld%lld%lld", &A, &B, &L);
        tree[A].push_back({B, L});
        tree[B].push_back({A, L});
    }
    dfs1(1, 0);
    dfs(1, 0);
    for (int i = 1; i <= N; i++) {
        minn = min(minn, dp[i]);
    }
    printf("%lld", minn);
}
相关推荐
vivo互联网技术16 小时前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦17 小时前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
clint45619 小时前
C++进阶(1)——前景提要
c++
用户4978630507320 小时前
(一)小红的数组操作
算法·编程语言
夜悊1 天前
C++代码示例:进制数简单生成工具
c++
怕浪猫1 天前
Electron 系列文章封面图
算法·架构·前端框架
郝学胜_神的一滴1 天前
CMake 021: IF 条件判据详诠
c++·cmake
徐小夕1 天前
JitWord 3.0 正式发布,高精度Word异构解析+复杂组件兼容,打造web端协同Word编辑器
前端·vue.js·算法
_wyt0012 天前
洛谷 B3930 [GESP202312 五级] 烹饪问题 题解
c++·gesp