颜色树

题目大意

树上每个节点均有颜色。
求有多少条路径包含了所有k种颜色。k<=10

点分治

点分治后处理到根的二进制状态表示包含的颜色。
问题转化成插入若干个二进制数,查询包含某个二进制数的二进制数有多少个。
可以插入O(1)查询O(2^k)
考虑综合插入和查询的复杂度。
设f[A,B]表示有多少二进制数前一半为A后一半包含B。
插入时A是确定的,枚举包含的B,O(2^(k/2))
查询时B是确定的,枚举被包含的A,O(2^(k/2))

#include<cstdio>
#include<algorithm>
#define fo(i,a,b) for(i=a;i<=b;i++)
using namespace std;
typedef long long ll;
const int maxn=50000+10;
int h[maxn],go[maxn*2],next[maxn*2];
int d[maxn],size[maxn],a[maxn],v[maxn],f[40][40];
bool bz[maxn];
int i,j,k,l,t,n,m,tot,top,all;
ll ans;
int read(){
    int x=0,f=1;
    char ch=getchar();
    while (ch<'0'||ch>'9'){
        if (ch=='-') f=-1;
        ch=getchar();
    }
    while (ch>='0'&&ch<='9'){
        x=x*10+ch-'0';
        ch=getchar();
    }
    return x*f;
}
void add(int x,int y){
    go[++tot]=y;
    next[tot]=h[x];
    h[x]=tot;
}
void travel(int x,int y){
    a[++top]=x;
    int t=h[x];
    size[x]=1;
    while (t){
        if (!bz[go[t]]&&go[t]!=y){
            travel(go[t],x);
            size[x]+=size[go[t]];
        }
        t=next[t];
    }
}
void dfs(int x,int y){
    int t=h[x];
    while (t){
        if (!bz[go[t]]&&go[t]!=y){
            d[go[t]]=d[x]|v[go[t]];
            dfs(go[t],x);
        }
        t=next[t];
    }
}
void calc(int sig){
    int i,j,k,l,t;
    fo(i,1,top){
        t=d[a[i]];
        j=t/32;k=l=t%32;
        while (1){
            f[j][k]++;
            if (!k) break;
            k=(k-1)&l;
        }
    }
    fo(i,1,top){
        t=d[a[i]]^all;
        j=l=t/32;k=t%32;
        j^=31;l^=31;
        while (1){
            ans+=(ll)f[j^31][k]*sig;
            if (!j) break;
            j=(j-1)&l;
        }
    }
    fo(i,1,top){
        t=d[a[i]];
        j=t/32;k=l=t%32;
        while (1){
            f[j][k]--;
            if (!k) break;
            k=(k-1)&l;
        }
    }
}
void solve(int x,int y){
    top=0;
    travel(x,0);
    if (y) calc(-1);
    int i,j=x,k=0,t;
    while (1){
        t=h[j];
        while (t){
            if (!bz[go[t]]&&go[t]!=k&&size[go[t]]>top/2){
                k=j;
                j=go[t];
                break;
            }
            t=next[t];
        }
        if (!t) break;
    }
    d[j]=v[j];
    dfs(j,0);
    calc(1);
    bz[j]=1;
    t=h[j];
    while (t){
        if (!bz[go[t]]) solve(go[t],1);
        t=next[t];
    }
}
int main(){
    freopen("colortree.in","r",stdin);freopen("colortree.out","w",stdout);
    n=read();m=read();
    all=(1<<m)-1;
    fo(i,1,n) v[i]=read(),v[i]=1<<(v[i]-1);
    fo(i,1,n-1){
        j=read();k=read();
        add(j,k);add(k,j);
    }
    solve(1,0);
    printf("%lld\n",ans);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值