Codeforces 1153D 树形DP

题意:有一个游戏,规则如下:每个点有一个标号,为max或min, max是指这个点的值是所有子节点中值最大的那一个,min同理。问如何给这颗树的叶子节点赋值,可以让这棵树的根节点值最大。

思路:很明显的树形dp, 设dp[x]是指以x为根的子树中可以获得的最大的值, sz[x]是指以x为根的子树中叶子节点的个数。

若x是max, 那么dp[x] = max(sz[x] - sz[y] + dp[y]),对应的决策相当于把最大的几个值给dp[y] - sz[y]最大的那颗子树。

若x是min, 首先需要统计每颗子树的sz[y] - dp[y], 这些都不可能被选上了,之后,还要统计有多少棵子树(假设有z棵), 其中z - 1个肯定选不上了,所以答案是sz[x] - Σ(sz[y] - dp[y]) - (z - 1)。

这两条对照这样例的图很容易就能发现了,然而昨晚思路跑偏了,这题都没做出来,含泪掉分。。。

代码:

#include <bits/stdc++.h>
using namespace std;
const int maxn = 300010;
vector<int> G[maxn];
int sz[maxn], dp[maxn], a[maxn];
void dfs(int x) {
	dp[x] = 1;
	if(G[x].size() == 0) {
		sz[x] = 1;
		return; 
	}
	for (int i = 0; i < G[x].size(); i++) {
		int y = G[x][i];
		dfs(y);
		sz[x] += sz[y];
	}
	if(a[x] == 1) {
		for (int i = 0; i < G[x].size(); i++) {
			int y = G[x][i];
			dp[x] = max(dp[x], sz[x] - sz[y] + dp[y]);
		}
	} else {
		int tmp = 0;
		for (int i = 0; i < G[x].size(); i++) {
			int y = G[x][i];
			tmp += sz[y] - dp[y];
		}
		dp[x] = sz[x] - tmp - G[x].size() + 1;
	}
}
int main() {
	int n, x;
	scanf("%d", &n);
	for (int i = 1; i <= n; i++) {
		scanf("%d", &a[i]);
	}
	for (int i = 2; i <= n; i++) {
		scanf("%d", &x);
		G[x].push_back(i);
	}
	dfs(1);
	printf("%d\n", dp[1]);
} 

  

posted @ 2019-04-14 12:17  维和战艇机  阅读(555)  评论(0编辑  收藏  举报