[BZOJ4027/HEOI2015]兔子与樱花
Description
很久很久之前,森林里住着一群兔子。有一天,兔子们突然决定要去看樱花。兔子们所在森林里的樱花树很特殊。樱花树由n个树枝分叉点组成,编号从0到n-1,这n个分叉点由n-1个树枝连接,我们可以把它看成一个有根树结构,其中0号节点是根节点。这个树的每个节点上都会有一些樱花,其中第i个节点有c_i朵樱花。樱花树的每一个节点都有最大的载重m,对于每一个节点i,它的儿子节点的个数和i节点上樱花个数之和不能超过m,即son(i) + c_i <= m,其中son(i)表示i的儿子的个数,如果i为叶子节点,则son(i) = 0。
Input
第一行输入两个正整数,n和m分别表示节点个数和最大载重;
Output
一行一个整数,表示最多能删除多少节点。
Sample Input
0 2 2 2 4 1 0 4 1 1
3 6 2 3
1 9
1 8
1 1
0
0
2 7 4
0
1 5
0
Sample Output
HINT
对于100%的数据,1 <= n <= 2000000, 1 <= m <= 100000, 0 <= c_i <= 1000
#include <cstdio>
#include <algorithm>
#define MAXN 2000005
using namespace std;
struct Edge { int v, next; } edge[MAXN];
int w[MAXN], son[MAXN], n, m, now, h[MAXN], v, f[MAXN], st[MAXN];
int read()
{
int res = 0; char ch = getchar();
while (ch < '0' && ch > '9') ch = getchar();
while (ch >= '0' && ch <= '9') res = res * 10 + ch - '0', ch = getchar();
return res;
}
int DFS(int o)
{
int cnt = 0;
for (int x = h[o]; x; x = edge[x].next)
{
int v = edge[x].v;
DFS(v), w[o]++, f[o] += f[v];
}
for (int x = h[o]; x; x = edge[x].next) st[++cnt] = w[edge[x].v];
sort(st + 1, st + cnt + 1);
for (int i = 1; i <= cnt; i++)
if (w[o] + st[i] - 1 <= m) w[o] += st[i] - 1, f[o]++; else break;
return f[o];
}
int main()
{
freopen("4027.in", "r", stdin);
freopen("4027.out", "w", stdout);
n = read(), m = read();
for (int i = 0; i <= n - 1; i++) w[i] = read();
for (int i = 0; i <= n - 1; i++)
{
son[i] = read();
for (int j = 1; j <= son[i]; j++) v = read(), now++, edge[now] = (Edge) {v, h[i]}, h[i] = now;
}
printf("%d", DFS(0));
return 0;
}