码迷,mamicode.com
首页 > 其他好文 > 详细

bzoj2589: Spoj 10707 Count on a tree II

时间:2016-08-07 00:41:52      阅读:361      评论:0      收藏:0      [点我收藏+]

标签:

Description

给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v),你需要回答u xor lastans和v这两个节点间有多少种不同的点权。其中lastans是上一个询问的答案,初始为0,即第一个询问的u是明文。

Input

第一行两个整数N,M。
第二行有N个整数,其中第i个整数表示点i的权值。
后面N-1行每行两个整数(x,y),表示点x到点y有一条边。
最后M行每行两个整数(u,v),表示一组询问。
数据范围是N<=40000 M<=100000 点权在int范围内 

Output

M行,表示每个询问的答案。
1.每次在树上找到一棵树高为sqrt(n)的子树分为一块并删去,最后剩下树高不足sqrt(n)的部分(如果有)另成一块,这样树就被分为至多sqrt(n)块,且每块高度不超过sqrt(n)
2.对每个块的树根,预处理出这个点到树上所有点的路径的答案
3.对每个点,预处理出这个点到1号点的路径上每种点权出现的最大深度,用可持久化块状数组保存(数组分为sqrt(n)块,每个节点保存指向这些块的指针,修改时暴力复制所修改的块并修改)
4.对一个询问(u,v),若u,v在同一块则暴力,否则令u所在块的根的深度大于v所在块的根的深度,设x为u所在块的根,则x到v的答案已预处理好,根据第3步预处理的内容可以查询u到x的路径上(不包括x)的点权是否在x到v的路径上出现过从而得到答案
以上每一步均是O(n3/2)时间复杂度,常数非常大,可能需要一些常数优化
#include<bits/stdc++.h>
const int N=40017;
int n,m,B;
int read(){
    int x=0,c=getchar();
    while(c<48)c=getchar();
    while(c>47)x=x*10+c-48,c=getchar();
    return x;
}
int v[N],vs[N],e[N*2][2],e0[N],ep=2,la=0;
int id[N],rt[N],idp=0,fa[N],dep[N],t[N],ANS=0;
int sz[N],pf[N],top[N];
int ans[210][N],h[N];
int mem[261*N],*ptr=mem;
struct Array{
    int*arr[161];
    const int&operator[](int x){
        return arr[x>>8][x&255];
    }
    void copy(Array&src,int x,int y){
        memcpy(&arr,&src.arr,161*4);
        ptr+=261;
        memcpy(ptr,arr[x>>8],256*4);
        ptr[x&255]=y;
        arr[x>>8]=ptr;
    }
}as[N];
int lca(int x,int y){
    int a=top[x],b=top[y];
    while(a!=b){
        if(dep[a]<dep[b])std::swap(a,b),std::swap(x,y);
        x=fa[a];a=top[x];
    }
    return dep[x]<dep[y]?x:y;
}
int vio(int x,int y){
    int a=lca(x,y),r=0;
    for(int w=x;w!=a;w=fa[w])if(!t[v[w]]++)++r;
    for(int w=y;w!=a;w=fa[w])if(!t[v[w]]++)++r;
    if(!t[v[a]])++r;
    for(int w=x;w!=a;w=fa[w])t[v[w]]=0;
    for(int w=y;w!=a;w=fa[w])t[v[w]]=0;
    return r;
}
int query(int x,int y){
    if(id[x]==id[y])return vio(x,y);
    if(dep[rt[id[x]]]<dep[rt[id[y]]])std::swap(x,y);
    int d=dep[lca(x,y)],b=rt[id[x]];
    int r=ans[id[b]][y];
    for(int w=x;w!=b;w=fa[w]){
        int c=v[w];
        if(!t[c]&&as[b][c]<d&&as[y][c]<d)++r,t[c]=1;
    }
    for(int w=x;w!=b;w=fa[w])t[v[w]]=0;
    return r;
}
void f3(int w,int pa,int ID){
    if(!t[v[w]]++)++ANS;
    ans[ID][w]=ANS;
    for(int i=e0[w];i;i=e[i][1]){
        int u=e[i][0];
        if(u!=pa)f3(u,w,ID);
    }
    if(!--t[v[w]])--ANS;
}
void f1(int w,int pa){
    sz[w]=1;
    fa[w]=pa;
    dep[w]=dep[pa]+1;
    for(int i=e0[w];i;i=e[i][1]){
        int u=e[i][0];
        if(u!=pa){
            f1(u,w);
            sz[w]+=sz[u];
            if(sz[u]>sz[pf[w]])pf[w]=u;
        }
    }
}
void f2(int w){
    for(int i=e0[w];i;i=e[i][1]){
        int u=e[i][0];
        if(u!=fa[w]&&!id[u]){
            id[u]=id[w];
            f2(u);
        }
    }
}
void f4(int w){
    as[w].copy(as[fa[w]],v[w],dep[w]);
    for(int i=e0[w];i;i=e[i][1]){
        int u=e[i][0];
        if(u!=fa[w])f4(u);
    }
}
void f5(int w,int tp){
    top[w]=tp;
    if(pf[w])f5(pf[w],tp);
    for(int i=e0[w];i;i=e[i][1]){
        int u=e[i][0];
        if(u!=fa[w]&&u!=pf[w])f5(u,u);
    }
}
void f6(int w){
    h[w]=0;
    for(int i=e0[w];i;i=e[i][1]){
        int u=e[i][0];
        if(u!=fa[w]&&!id[u]){
            f6(u);
            if(h[u]>=h[w])h[w]=h[u]+1;
        }
    }
}int main(){
    n=read();m=read();
    B=sqrt(n+1)+1;
    for(int i=1;i<=n;i++)vs[i]=v[i]=read();
    std::sort(vs+1,vs+n+1);
    for(int i=1;i<=n;i++)v[i]=std::lower_bound(vs+1,vs+n+1,v[i])-vs;
    for(int i=1;i<n;i++){
        int a=read(),b=read();
        e[ep][0]=b;e[ep][1]=e0[a];e0[a]=ep++;
        e[ep][0]=a;e[ep][1]=e0[b];e0[b]=ep++;
    }
    for(int i=0;i<161;i++)as[0].arr[i]=mem;
    f1(1,0);
    do{
        f6(1);
        int r=1;
        for(int i=1;i<=n;i++)if(!id[i]&&h[i]==B){
            r=i;
            break;
        }
        rt[id[r]=++idp]=r;
        f2(r);
        f3(r,0,idp);
    }while(!id[1]);
    f4(1);f5(1,1);
    while(m--){
        int x=read()^la,y=read();
        printf("%d\n",la=query(x,y));
    }
    return 0;
}

 

bzoj2589: Spoj 10707 Count on a tree II

标签:

原文地址:http://www.cnblogs.com/ccz181078/p/5745206.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!