题目
某大学有N个职员,编号为1~N。他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司。现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数Ri,但是呢,如果某个职员的上司来参加舞会了,那么这个职员就无论如何也不肯来参加舞会了。所以,请你编程计算,邀请哪些职员可以使快乐指数最大,求最大的快乐指数。
输入格式:
第一行一个整数N。(1<=N<=6000)
接下来N行,第i+1行表示i号职员的快乐指数Ri。(-128<=Ri<=127)
接下来N-1行,每行输入一对整数L,K。表示K是L的直接上司。
最后一行输入0 0
输出格式:
输出最大的快乐指数。
分析
一道经典的树型dp模板题,dp[i][0/1]表示第i个节点选还是不选,因为如果一个点选了它的儿子都不能选,一个点如果不选它的儿子既可以选也可以不选,所以我们不难得出代码中的转移方程
代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<cmath>
#include<cstdlib>
#include<algorithm>
#include<queue>
#include<stack>
#include<map>
#include<vector>
using namespace std;
int n,m;
vector<int>v[6010];
int dp[6500][3];
int val[6500];
void dfs(int x,int fa){
int i;
dp[x][1]=val[x];
if(v[x].size()==0)return;
for(i=0;i<v[x].size();i++)
if(v[x][i]!=fa){
dfs(v[x][i],x);
dp[x][0]+=max(0,max(dp[v[x][i]][0],dp[v[x][i]][1]));
dp[x][1]+=max(0,dp[v[x][0]][0]);
}
return;
}
int main(){
int i,j,k;
cin>>n;
for(i=1;i<=n;i++)
cin>>val[i];
for(i=1;i<n;i++){
int x,y;
cin>>x>>y;
v[x].push_back(y);
v[y].push_back(x);
}
cin>>i>>j;
dfs(1,0);
cout<<max(dp[1][0],dp[1][1])<<endl;
return 0;
}