没有上司的舞会(树形dp)

题目:
Ural大学有N名职员,编号为1~N。
他们的关系就像一棵以校长为根的树,父节点就是子节点的直接上司。
每个职员有一个快乐指数,用整数 HiHi 给出,其中 1≤i≤N,1≤i≤N1\leq i\leq N,1\leq i\leq N1≤i≤N,1≤i≤N。
现在要召开一场周年庆宴会,不过,没有职员愿意和直接上司一起参会。
在满足这个条件的前提下,主办方希望邀请一部分职员参会,使得所有参会职员的快乐指数总和最大,求这个最大值。

输入:
第一行一个整数N。
接下来N行,第 i 行表示 i 号职员的快乐指数Hi。
接下来N-1行,每行输入一对整数L, K,表示K是L的直接上司。
最后一行输入0,0。

输出:

输出最大的快乐指数

示例:

输入:

7
1
1
1
1
1
1
1
1 3
2 3
6 4
7 4
4 5
3 5
0 0

输出:

5

备注:

 

-128\leqslant H_{i}\leqslant 127

思路:

我们将这道题以树状存储,如果一个节点代表的职员参加宴会,那么它的子节点代表的职员就不能参加宴会。我们将dp[i][0]表示不选择当前节点代表职员参加宴会的最优解,dp[i][1]表示选择当前节点参加的最优解。

假设v为u的子节点。那么可以得到:

①dp[u][0] += max(dp[v][0],dp[v][1])

②dp[u][1] += dp[v][0]

代码:

#include <bits/stdc++.h>
using namespace std;

const int N = 6005;
int v[N],dp[N][2],fa[N];
vector <int> G[N];

void edge(int a,int b) {
    G[a].push_back(b);
    fa[b] = a;//父节点
}

void dfs(int u) {
    dp[u][0] = 0;
    dp[u][1] = v[u];
    for(int i = 0; i < G[u].size(); i++) {
        dfs(G[u][i]);
        dp[u][1] += dp[G[u][i]][0];
        dp[u][0] += max(dp[G[u][i]][0],dp[G[u][i]][1]);
    }
}


int main()
{
    int n;
    cin >> n;
    for(int i = 1; i <= n; i++) {
        cin >> v[i];
    }
    for(int i = 1; i < n; i++) {
        int a,b;
        cin >> a >> b;
        edge(b,a);
    }
    //寻找根节点
    int t = 1;
    while(fa[t]) {
        t = fa[t];
    }
    dfs(t);
    cout << max(dp[t][0],dp[t][1]) <<endl;
    return 0;
}

  • 12
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值