P3177 树上染色做题记录

树形 dp 好题。

做这题的思想历程:

定义 \(dp_{i,j}\) 表示以 \(i\) 为根的子树中,选择了 \(j\) 个节点的答案。感觉还要带上一维状态就是所有黑点距离 \(i\) 的距离,这违反了做题思路中间的简洁性的原则。于是我们 查看题解

经过不明方法之后,我们想到了定义 \(dp_{i,j}\) 对于答案的总贡献最小是多少。为什么我们能算出全局贡献,因为我们既然知道了有多少个黑点有多少个白点,那么我们就知道了一条边的左右有多少个黑白点,那么这样的话我们可以尝试一下背包转移就完事儿了 /jy

果然解决了,直接背包就完事儿了。。

#include <bits/stdc++.h>
using namespace std;
template <typename T>inline void read(T& t){t=0; register char ch=getchar(); register int fflag=1;while(!('0'<=ch&&ch<='9')) {if(ch=='-') fflag=-1;ch=getchar();}while(('0'<=ch&&ch<='9')){t=t*10+ch-'0'; ch=getchar();} t*=fflag;}
template <typename T,typename... Args> inline void read(T& t, Args&... args) {read(t);read(args...);}
const int N = 2000 + 10, inf = 0x3f3f3f3f;

int sz[N], n, k;
typedef long long ll;
ll dp[N][N], tmp[N << 2];
vector<pair<int, int> >G[N];

void chkmax(ll &x, ll y) {x = (y > x) ? y : x;}

void dfs(int u, int fa) {
    sz[u] = 1;
    for(auto path:G[u]) {
        int v = path.first, val = path.second;
        if(v == fa) continue;
        dfs(v, u);
        for(int i = 0; i <= sz[u] + sz[v]; ++i) tmp[i] = 0;
        for(int i = 0; i <= sz[u]; ++i)
            for(int j = 0; j <= sz[v]; ++j) {
                chkmax(tmp[i + j], dp[u][i] + dp[v][j] + 1ll * j * (k - j) * val /*black*/ + 1ll * (sz[v] - j) * (n - k - (sz[v] - j)) * val /*white*/);
            }
        sz[u] += sz[v];
        for(int i = 0; i <= sz[u]; ++i) dp[u][i] = tmp[i];
    }
}

int main() {
    read(n, k);
    for(int i = 1; i < n; ++i) {
        int x, y, val;
        read(x, y, val);
        G[x].push_back({y, val});
        G[y].push_back({x, val});
    }
    dfs(1, -1);
    cout << dp[1][k] << endl;
    return 0;
}
posted @ 2022-09-17 07:02  Mercury_City  阅读(27)  评论(0编辑  收藏  举报