首页 > 代码库 > ural1018(树形dp)
ural1018(树形dp)
题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=17662
题意:给一棵边有权值的二叉树,节点编号为1~n,1是根节点。求砍掉一些边,只保留q条边,这q条边构成的子树的根节点要求是1,求这颗子树的最大权值。
分析:1.dp[u][i]表示已以u为根节点的子树,保留了i个节点的最大权值。每条边的权值,把它看作是连接的两个节点中的儿子节点的权值。那么总共要保留m+1个节点。
所以有:dp[u][i]=max(dp[u][i],dp[v][k]+dp[u][i-k]+w).ans=dp[1][m+1].
2.dp[u][i]表示u为根节点的子树,保留了i个条边的最大权值。在以u点为根节点的子树中选择了k条边,那么u的儿子节点v代表的子树中至多选k-1条边,因为如果在v子树中选边,那么u到v的边必选。
所以有:dp[u][i]=max(dp[u][i],dp[u][i-k]+dp[v][k-1]+w).ans=dp[1][m].
方法1的代码:
#include <cstdio>#include <cstring>#include <cmath>#include <iostream>#include <algorithm>#include <queue>#include <cstdlib>#include <stack>#include <vector>#include <set>#include <map>#define LL long long#define mod 1000000007#define inf 0x3f3f3f3f#define N 110#define clr(a) (memset(a,0,sizeof(a)))using namespace std;struct edge{ int v,w,next; edge(){} edge(int v,int w,int next):v(v),w(w),next(next){}}e[2*N];int head[N],dp[N][N],num[N],tot,n,m;void addedge(int u,int v,int w){ e[tot]=edge(v,w,head[u]); head[u]=tot++;}void dfs(int u,int fa){ num[u]=1; for(int i=head[u];~i;i=e[i].next) { int v=e[i].v,w=e[i].w; if(v==fa)continue; dfs(v,u); num[u]+=num[v]; for(int j=num[u];j>=1;j--) for(int k=1;k<j&&k<=num[v];k++)//注意这里k<j.因为子树根节点u点必选,在子树节点中至多选j-1个 dp[u][j]=max(dp[u][j],dp[u][j-k]+dp[v][k]+w); }}int main(){ int u,v,w; while(scanf("%d%d",&n,&m)>0) { memset(head,-1,sizeof(head)); clr(dp);tot=0; for(int i=1;i<n;i++) { scanf("%d%d%d",&u,&v,&w); addedge(u,v,w); addedge(v,u,w); } dfs(1,-1); printf("%d\n",dp[1][m+1]); }}
方法2的代码:
#include <cstdio>#include <cstring>#include <cmath>#include <iostream>#include <algorithm>#include <queue>#include <cstdlib>#include <stack>#include <vector>#include <set>#include <map>#define LL long long#define mod 1000000007#define inf 0x3f3f3f3f#define N 110#define clr(a) (memset(a,0,sizeof(a)))using namespace std;struct edge{ int v,w,next; edge(){} edge(int v,int w,int next):v(v),w(w),next(next){}}e[2*N];int head[N],dp[N][N],tot,n,m;void addedge(int u,int v,int w){ e[tot]=edge(v,w,head[u]); head[u]=tot++;}void dfs(int u,int fa){ for(int i=head[u];~i;i=e[i].next) { int v=e[i].v,w=e[i].w; if(v==fa)continue; dfs(v,u); for(int j=m;j>=1;j--) for(int k=1;k<=j;k++) dp[u][j]=max(dp[u][j],dp[u][j-k]+dp[v][k-1]+w); }}int main(){ int u,v,w; while(scanf("%d%d",&n,&m)>0) { memset(head,-1,sizeof(head)); clr(dp);tot=0; for(int i=1;i<n;i++) { scanf("%d%d%d",&u,&v,&w); addedge(u,v,w); addedge(v,u,w); } dfs(1,-1); printf("%d\n",dp[1][m]); }}
ural1018(树形dp)
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。