概述
[CTSC1997]选课
【树形
DP
text{DP}
DP】
一眼算法题(区分差分约束与拓扑排序)
注意到有些功课可能没有前置功课,于是把所有没有前置功课的点都与
0
text{0}
0连边,然后进行树形
DP
text{DP}
DP。注意到
0
text{0}
0这个点要强制被选,那么最后选课个数就是
m+1
text{m+1}
m+1。注意这道题属于
01
text{01}
01背包类树形
DP
text{DP}
DP,以节点编号作为
DP
text{DP}
DP阶段,像线性
DP
text{DP}
DP一样,把当前背包的体积作为第二维状态,在状态转移的时候实际上就是一个分组背包问题。
#include <bits/stdc++.h>
using namespace std;
const int N=350;
int n,m,x,y,ans,f[N][N],sz[N],fir[N<<1],nxt[N<<1],to[N<<1],tot;
inline int read(){
int cnt=0,f=1;char c=getchar();
while(!isdigit(c)){if(c=='-')f=-f;c=getchar();}
while(isdigit(c)){cnt=(cnt<<1)+(cnt<<3)+(c^48);c=getchar();}
return cnt*f;
}
inline void add(int x,int y){nxt[++tot]=fir[x];fir[x]=tot;to[tot]=y;}
void dfs(int u){
for(int i=fir[u];i;i=nxt[i]){
int v=to[i];
dfs(v);sz[u]+=sz[v]+1;
for(int j=max(m+1,sz[u]);j;--j){//倒序循环正确处理组内体积为0的物品
for(int k=0;k<=j-1;++k){
f[u][j]=max(f[u][j],f[u][j-k]+f[v][k]);
}
}
}
}
signed main(){
n=read(),m=read();
for(int i=1;i<=n;++i) {x=read(),f[i][1]=read(),add(x,i);}
dfs(0);
printf("%d",f[0][m+1]);
return 0;
}
最后
以上就是俭朴小鸭子为你收集整理的[CTSC1997]选课【背包类树形dp】的全部内容,希望文章能够帮你解决[CTSC1997]选课【背包类树形dp】所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复