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;
}
标签:int,染色,ll,P3177,read,path,树上,void,dp 来源: https://www.cnblogs.com/Mercury-City/p/16701263.html