【CF486D】有效集合-树形dp

Problem 有效集合

题目大意

给出一棵树,求出这棵树的不同联通子节点集合的数量,这些集合必须满足最大权值点减最小权值点小于等于d。

Solution

再一次树d乱搞。

因为数据范围贼小,所以我们对于每一个点为根的情况进行一次dfs.

对于以节点root为根的情况,我们认为root是最小值。

对于root的某一棵子树,其需要算的情况一定包含root,否则前面已经算过了。

于是我们就对这个root进行dfs。dfs过程中,我们始终保证root为最小值,不遍历大于它的点。

因为有了这个性质,我们只要不走加起来大于d的点,接下来所有的点都不属于被计算范围内,可以不走。

在这个以root为根的树中,现在我们遍历到u,选中它的一颗子树v。

对于这一串东西上,我们每次可以选择以前已遍历过的子树中选择某些点,再从v该棵子树下选择某些点组成新情况。

我们增加的情况是已经遍历过的子树答案*(该子树被遍历到的子树总答案+1)

加一是因为空集也算是一种情况。

也就是f[u]*=f[v]+1;

对于每个叶子结点,我们只需要将f设为1即可。

dfs完以后,我们将以每个点为根(最小值)的情况进行累加。

于是最终的ans便是答案。

 

需要注意的是,这一道题MOD1000000007

所以f数组、ans变量需要long long类型

而我一直爆炸,因为没注意到,dfs函数也要为long long类型。

所以请注意这个问题。

AC Code

#include <iostream>
#include <cstdio>
#include <cstring>
#define MOD 1000000007
using namespace std;
struct node{
    int to,next;
}e[4010];
int d,n,u,v,h[2010],a[2010],tot=0;
long long f[2010],ans=0;
void add(int u,int v){
    e[++tot].to=u;e[tot].next=h[v];h[v]=tot;
    e[++tot].to=v;e[tot].next=h[u];h[u]=tot;
}
long long dfs(int x,int last,int root){
    long long ans=1;
    f[x]=1;
    for(int i=h[x];~i;i=e[i].next){
        if(last!=e[i].to&&(a[e[i].to]>a[root]||(a[e[i].to]==a[root]&&root>e[i].to))
        &&a[e[i].to]-a[root]<=d){
            ans=(ans*(dfs(e[i].to,x,root)+1)%MOD)%MOD;
        }
    }
    return ans;
}
int main(){
//  freopen("cf486d.in","r",stdin);
    memset(h,-1,sizeof(h));
    scanf("%d%d",&d,&n);
    for(int i=1;i<=n;i++)scanf("%d",&a[i]);
    for(int i=1;i<n;i++){
        scanf("%d%d",&u,&v);
        add(u,v);
    }
    for(int i=1;i<=n;i++)ans+=dfs(i,0,i);
    printf("%lld",ans%MOD);
}

 

posted @ 2017-07-09 22:10  skylynf  阅读(543)  评论(0编辑  收藏  举报