【bzoj2060】[Usaco2010 Nov]Visiting Cows拜访奶牛 树形dp

题目描述

经过了几周的辛苦工作,贝茜终于迎来了一个假期.作为奶牛群中最会社交的牛,她希望去拜访N(1<=N<=50000)个朋友.这些朋友被标号为1..N.这些奶牛有一个不同寻常的交通系统,里面有N-1条路,每条路连接了一对编号为C1和C2的奶牛(1 <= C1 <= N; 1 <= C2 <= N; C1<>C2).这样,在每一对奶牛之间都有一条唯一的通路. FJ希望贝茜尽快的回到农场.于是,他就指示贝茜,如果对于一条路直接相连的两个奶牛,贝茜只能拜访其中的一个.当然,贝茜希望她的假期越长越好,所以她想知道她可以拜访的奶牛的最大数目.

输入

1行:单独的一个整数N

2..N行:每一行两个整数,代表了一条路的C1和C2.

输出

单独的一个整数,代表了贝茜可以拜访的奶牛的最大数目.

样例输入

7

6 2

3 4

2 3

1 2

7 6

5 6

样例输出

4


题解

裸的树形dp。

f[x]代表拜访x时最大数量,g[x]代表不拜访x时最大数量。

那么易推得f[x]=1+∑g[to[i]],g[x]=∑max(f[to[i]],g[to[i]])。

答案即为max(f[1],g[1])。

#include <stdio.h>
#include <string.h>
int to[100001] , next[100001] , head[50001] , f[50001] , g[50001] , cnt;
int max(int a , int b)
{
    return a > b ? a : b;
}
void add(int x , int y)
{
    to[cnt] = y;
    next[cnt] = head[x];
    head[x] = cnt ++ ;
}
void dp(int x , int last)
{
    int i , y;
    f[x] = 1;
    for(i = head[x] ; i != -1 ; i = next[i])
    {
        y = to[i];
        if(y == last)
            continue;
        dp(y , x);
        f[x] += g[y];
        g[x] += max(f[y] , g[y]);
    }
}
int main()
{
    int n , i , x , y;
    scanf("%d" , &n);
    memset(head , -1 , sizeof(head));
    for(i = 1 ; i <= n - 1 ; i ++ )
    {
        scanf("%d%d" , &x , &y);
        add(x , y);
        add(y , x);
    }
    dp(1 , 0);
    printf("%d\n" , max(f[1] , g[1]));
    return 0;
}
posted @ 2016-12-13 20:09  GXZlegend  阅读(467)  评论(0编辑  收藏  举报