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