B20J_4027_[HEOI2015]兔子与樱花_树形DP

B20J_4027_[HEOI2015]兔子与樱花_树形DP

题意:

很久很久之前,森林里住着一群兔子。有一天,兔子们突然决定要去看樱花。兔子们所在森林里的樱花树很特殊。樱花树由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

现在兔子们觉得樱花树上节点太多,希望去掉一些节点。当一个节点被去掉之后,这个节点上的樱花和它的儿子节点都被连到删掉节点的父节点上。如果父节点也被删除,那么就会继续向上连接,直到第一个没有被删除的节点为止。
现在兔子们希望计算在不违背最大载重的情况下,最多能删除多少节点。
注意根节点不能被删除,被删除的节点不被计入载重。
 
分析:
我们仔细思考一下就会发现从下往上删不会使答案变差,并且删儿子不会对父亲的祖先产生影响。
对于每个结点,贪心的取儿子中贡献小的,并加入到自身的贡献
一个点的贡献 = 儿子的个数 + 这个点的权值
 
代码:
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 2000050
int head[N],to[N<<1],nxt[N<<1],cnt;
int n,m,c[N],son[N],a[N],tot,b[N],ans,nm[N];
inline void add(int u,int v){
    to[++cnt]=v;nxt[cnt]=head[u];head[u]=cnt;
}
void read(int &x){
    int f=1;x=0;char s=getchar();
    while(s<'0'||s>'9'){if(s=='-')f=-1;s=getchar();}
    while(s>='0'&&s<='9'){x=(x<<3)+(x<<1)+s-'0';s=getchar();}
    x*=f;
}
void dfs(int x,int y){
    if(son[x]==0)return ;
    for(int i=head[x];i;i=nxt[i]){
        if(to[i]!=y){
            dfs(to[i],x);
        }
    }
    int cnt=0;
    for(int i=head[x];i;i=nxt[i]){
        if(to[i]!=y){
            b[++cnt]=c[to[i]];
        }
    }
    sort(b+1,b+cnt+1);
    c[x]+=son[x];
    for(int i=1;i<=cnt;i++){
        if(b[i]+c[x]-1<=m){
            c[x]+=b[i]-1;ans++;
        }else break;
    }
}
int main(){
    read(n);read(m);
    for(int i=1;i<=n;i++){
        read(c[i]);
    }
    int x,y;
    for(int i=1;i<=n;i++){
        read(x);son[i]=x;
        while(x--){
            read(y);y++;
            add(i,y);add(y,i);
        }
    }
    dfs(1,0);
    printf("%d",ans);
}

 

posted @ 2018-03-04 22:30  fcwww  阅读(170)  评论(0编辑  收藏  举报