Codeforces 1240C. Paint the Tree
首先每个点 $u$ 只能选择不超过 $k$ 个相连的边
并且设边为 $(u,v)$ ,那么此时 $v$ 也必须选择这条边
因为图是一颗树,显然考虑一下树形 $dp$
设 $f[x][0/1]$ 表示考虑完 $x$ 的子树,当前节点有没有留一个选择给和父亲相连的边($0$ 表示没有)
那么对于 $f[x][0]$
考虑所有 $x$ 的儿子 $v$,我们要选出不超过 $k$ 个儿子的 $f[v][1]+val(x,v)$ ,然后剩下的儿子全部选 $f[v][0]$ ,求最大价值
(其中 $val(x,v)$ 是边 $(x,v)$ 的价值)
考虑一开始所有的儿子都先选 $f[v][0]$,对于某个儿子 $v$ 如果我们之后要选 $f[v][1]$ ,那么增加的贡献 $delta$ 为 $-f[v][0]+f[v][1]+val(x,v)$
显然 $-f[v][0]$ 是因为之前已经加入了 $f[v][0]$ 的贡献
那么此时每个儿子的选择互不影响,直接按 $delta$ 排序取前 $k$ 大即可(注意如果还没到 $k$ 个但是 $delta$ 已经小于 $0$ 了就不用选)
对于 $f[x][1]$ 也是一样的道理,但是我们这时候取的就是前 $k-1$ 的的 $delta$ 了
不妨设 $1$ 为根,那么答案即为 $f[1][0]$
#include<iostream> #include<cstdio> #include<algorithm> #include<cstring> #include<cmath> #include<vector> using namespace std; typedef long long ll; inline int read() { int x=0,f=1; char ch=getchar(); while(ch<'0'||ch>'9') { if(ch=='-') f=-1; ch=getchar(); } while(ch>='0'&&ch<='9') { x=(x<<1)+(x<<3)+(ch^48); ch=getchar(); } return x*f; } const int N=5e5+7; int Q,n,K; int fir[N],from[N<<1],to[N<<1],val[N<<1],cntt; inline void add(int a,int b,int c) { from[++cntt]=fir[a]; fir[a]=cntt; to[cntt]=b; val[cntt]=c; } ll f[N][2]; void dfs(int x,int fa) { vector <ll> tmp; for(int i=fir[x];i;i=from[i]) { int v=to[i]; if(v==fa) continue; dfs(v,x); f[x][0]+=f[v][0]; f[x][1]+=f[v][0]; tmp.push_back(f[v][1]-f[v][0]+val[i]); } sort(tmp.begin(),tmp.end()); reverse(tmp.begin(),tmp.end()); int sz=tmp.size(); for(int i=0;i<K&&i<sz;i++) { if(tmp[i]<=0) break; f[x][0]+=tmp[i]; if(i<K-1) f[x][1]+=tmp[i]; } } int main() { Q=read(); while(Q--) { n=read(),K=read(); int a,b,c; for(int i=1;i<=n;i++) f[i][0]=f[i][1]=0; for(int i=1;i<=n;i++) fir[i]=0; cntt=0; for(int i=1;i<n;i++) { a=read(),b=read(),c=read(); add(a,b,c); add(b,a,c); } dfs(1,0); printf("%lld\n",f[1][0]); } return 0; }