• CF1324F Maximum White Subtree 题解 树形DP


    题目链接:https://www.luogu.com.cn/problem/CF1324F

    解题思路:
    两边dfs,第一遍求 (f1[i]) 表示以 (i) 为根节点并且包含 (i) 的最大值;第二遍求 (f2[i]) 表示 (i) 往父节点走(且不包含 (i) 的最大值)((f2[i] ge 0),如果 (f2[i] == 0) 可以理解为 (i) 上头的那些都不要了)。

    实现代码如下:

    #include <bits/stdc++.h>
    using namespace std;
    const int maxn = 200020;
    int n, a[maxn], f1[maxn], pa[maxn], f2[maxn];
    vector<int> g[maxn];
    void dfs1(int u) {
        f1[u] = a[u];
        int sz = g[u].size();
        for (int i = 0; i < sz; i ++) {
            int v = g[u][i];
            if (v == pa[u]) continue;
            pa[v] = u;
            dfs1(v);
            if (f1[v] > 0) f1[u] += f1[v];
        }
    }
    void dfs2(int u) {
        f2[u] = 0;
        if (pa[u]) {
            int pp = pa[u];
            if (f1[u] > 0) f2[u] = f1[pp] + f2[pp] - f1[u];
            else f2[u] = f1[pp] + f2[pp];
            if (f2[u] < 0) f2[u] = 0;
        }
        int sz = g[u].size();
        for (int i = 0; i < sz; i ++) {
            int v = g[u][i];
            if (v == pa[u]) continue;
            dfs2(v);
        }
    }
    int main() {
        scanf("%d", &n);
        for (int i = 1; i <= n; i ++) {
            scanf("%d", a+i);
            if (!a[i]) a[i] = -1;
        }
        for (int i = 1; i < n; i ++) {
            int u, v;
            scanf("%d%d", &u, &v);
            g[u].push_back(v);
            g[v].push_back(u);
        }
        dfs1(1);
        dfs2(1);
        for (int i = 1; i <= n; i ++) {
            if (i > 1) putchar(' ');
            int res = f1[i];
            if (f2[i] > 0) res += f2[i];
            printf("%d", res);
        }
        puts("");
        return 0;
    }
    
  • 相关阅读:
    2019.8.30 玉米田
    2019暑假集训 最短路计数
    2019暑假集训 旅行计划
    2019暑假集训 文件压缩
    0033-数字和星期转换
    0032-分数等级转换
    0031-闰年判断
    0030-购买衣服
    0029-求最小的数
    0028-判断奇偶
  • 原文地址:https://www.cnblogs.com/quanjun/p/12486440.html
Copyright © 2020-2023  润新知