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)时间复杂度,常数非常大,可能需要一些常数优化
upd: (1)中分块可在O(n)时间完成,另外bitset优化的 树分块+ST表(同bzoj4763)可以做到n2/32,且实际运行效果比上述算法更好
#include<bits/stdc++.h>
const int N=;
int n,m,B;
int read(){
int x=,c=getchar();
while(c<)c=getchar();
while(c>)x=x*+c-,c=getchar();
return x;
}
int v[N],vs[N],e[N*][],e0[N],ep=,la=;
int id[N],rt[N],idp=,fa[N],dep[N],t[N],ANS=;
int sz[N],pf[N],top[N];
int ans[][N],h[N];
int mem[*N],*ptr=mem;
struct Array{
int*arr[];
const int&operator[](int x){
return arr[x>>][x&];
}
void copy(Array&src,int x,int y){
memcpy(&arr,&src.arr,*);
ptr+=;
memcpy(ptr,arr[x>>],*);
ptr[x&]=y;
arr[x>>]=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=;
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]]=;
for(int w=y;w!=a;w=fa[w])t[v[w]]=;
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]=;
}
for(int w=x;w!=b;w=fa[w])t[v[w]]=;
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][]){
int u=e[i][];
if(u!=pa)f3(u,w,ID);
}
if(!--t[v[w]])--ANS;
}
void f1(int w,int pa){
sz[w]=;
fa[w]=pa;
dep[w]=dep[pa]+;
for(int i=e0[w];i;i=e[i][]){
int u=e[i][];
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][]){
int u=e[i][];
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][]){
int u=e[i][];
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][]){
int u=e[i][];
if(u!=fa[w]&&u!=pf[w])f5(u,u);
}
}
void f6(int w){
h[w]=;
for(int i=e0[w];i;i=e[i][]){
int u=e[i][];
if(u!=fa[w]&&!id[u]){
f6(u);
if(h[u]>=h[w])h[w]=h[u]+;
}
}
}int main(){
n=read();m=read();
B=sqrt(n+)+;
for(int i=;i<=n;i++)vs[i]=v[i]=read();
std::sort(vs+,vs+n+);
for(int i=;i<=n;i++)v[i]=std::lower_bound(vs+,vs+n+,v[i])-vs;
for(int i=;i<n;i++){
int a=read(),b=read();
e[ep][]=b;e[ep][]=e0[a];e0[a]=ep++;
e[ep][]=a;e[ep][]=e0[b];e0[b]=ep++;
}
for(int i=;i<;i++)as[].arr[i]=mem;
f1(,);
do{
f6();
int r=;
for(int i=;i<=n;i++)if(!id[i]&&h[i]==B){
r=i;
break;
}
rt[id[r]=++idp]=r;
f2(r);
f3(r,,idp);
}while(!id[]);
f4();f5(,);
while(m--){
int x=read()^la,y=read();
printf("%d\n",la=query(x,y));
}
return ;
}

bzoj2589: Spoj 10707 Count on a tree II的更多相关文章

  1. 【BZOJ2589】 Spoj 10707 Count on a tree II

    BZOJ2589 Spoj 10707 Count on a tree II Solution 吐槽:这道题目简直...丧心病狂 如果没有强制在线不就是树上莫队入门题? 如果加了强制在线怎么做? 考虑 ...

  2. BZOJ2539 Spoj 10707 Count on a tree II

    题面 题解 因为这道题目我也不太会做,所以借鉴了一下大佬heyujun的博客 如果不强制在线,这道题目是树上莫队练手题 我们知道莫队是离线的,但是万一强制在线就凉凉了 于是我们就需要一些操作:树分块 ...

  3. [BZOJ]2589: Spoj 10707 Count on a tree II

    Time Limit: 20 Sec  Memory Limit: 400 MB Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v),你需要回答u xor last ...

  4. 【BZOJ2589】[SPOJ10707]Count on a tree II

    [BZOJ2589][SPOJ10707]Count on a tree II 题面 bzoj 题解 这题如果不强制在线就是一个很\(sb\)的莫队了,但是它强制在线啊\(qaq\) 所以我们就用到了 ...

  5. 【SPOJ】Count On A Tree II(树上莫队)

    [SPOJ]Count On A Tree II(树上莫队) 题面 洛谷 Vjudge 洛谷上有翻译啦 题解 如果不在树上就是一个很裸很裸的莫队 现在在树上,就是一个很裸很裸的树上莫队啦. #incl ...

  6. spoj COT2 - Count on a tree II

    COT2 - Count on a tree II http://www.spoj.com/problems/COT2/ #tree You are given a tree with N nodes ...

  7. SPOJ COT2 - Count on a tree II(LCA+离散化+树上莫队)

    COT2 - Count on a tree II #tree You are given a tree with N nodes. The tree nodes are numbered from  ...

  8. SPOJ COT2 Count on a tree II(树上莫队)

    题目链接:http://www.spoj.com/problems/COT2/ You are given a tree with N nodes.The tree nodes are numbere ...

  9. SPOJ COT2 Count on a tree II (树上莫队)

    题目链接:http://www.spoj.com/problems/COT2/ 参考博客:http://www.cnblogs.com/xcw0754/p/4763804.html上面这个人推导部分写 ...

随机推荐

  1. OpenHCI - 4.2 Endpoint Descriptor

    4.2 Endpoint DescriptorAn Endpoint Descriptor (ED) is a 16-byte, memory resident structure that must ...

  2. java访问webservce,保持会话,服务端保存session验证

    在进行程序开发的过程中,遇到一个问题,怎么保持会话. 因为一帮进行方法调用很少涉及到即时身份验证的. 例如: 1:客户端登录后服务端保存登录用户信息: 2:客户端持有验证通过key再次请求: 3:服务 ...

  3. 一张图让你学会LVM

    导读 随着科技的进步,人们不知不觉的就进入了大数据的时代,数据的不断增加我们发现我们的磁盘越来越不够用了,接下来就是令人头疼的事情--加硬盘,数据的备份与还原.LVM就是Linux下专门针对我们数据的 ...

  4. HDU 4311 前缀和

    Description It has been ten years since TJU-ACM established. And in this year all the retired TJU-AC ...

  5. 百度APIStore

    链接 http://apistore.baidu.com/ 提供了许多免费的api接口,用来做功能性的查询

  6. Codeforces Round #150 (Div. 2)

    A. Dividing Orange 模拟. B. Undoubtedly Lucky Numbers 暴力枚举\(x.y\). C. The Brand New Function 固定左端点,右端点 ...

  7. JAVA常用关键字

    Java 中常用关键字: 一一解释(先以印象注明含义,若有错误或未填写的待用到后补充.更新):(蓝色为不确定部分) abstract : 虚类 boolean : 类型定义——布尔型 break : ...

  8. CentOS 6.0修改ssh远程连接端口

    转自:系统运维 » CentOS 6.0修改ssh远程连接端口 实现目的:把ssh默认远程连接端口修改为2222 方法如下: 1.编辑防火墙配置:vi /etc/sysconfig/iptables ...

  9. MFC如何添加自定义控件

    project->add to project->components and controls在Registered ActiveX Controls下找到你注册的自定义控件就可以了   ...

  10. css2图片边框

    用父元素的背景作为边框图片 父元素的背景既可以是不平铺的成品边框图片,也可以是平铺的图片,填充子元素和父元素之间的padding 然后给子元素设置背景色或背景图覆盖掉父元素的背景图.