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;
}
posted @ 2020-03-13 14:51  quanjun  阅读(116)  评论(0编辑  收藏  举报