题目描述

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

输入

第一行两个整数N,M。
第二行有N个整数,其中第i个整数表示点i的权值。
后面N-1行每行两个整数(x,y),表示点x到点y有一条边。
最后M行每行两个整数(u,v,k),表示一组询问。

输出

M行,表示每个询问的答案。

样例输入

8 5
105 2 9 3 8 5 7 7
1 2
1 3
1 4
3 5
3 6
3 7
4 8
2 5 1
0 5 2
10 5 3
11 5 4
110 8 2

样例输出

2
8
9
105
7


题解

主席树+最近公共祖先

需要明确主席树的原理:线段树相加减。

那么A到B的路径就是 A到根的路径+B到根的路径-最近公共祖先到根的路径-最近公共祖先的父亲到根的路径。

可以直接在树上建立主席树,注意每棵树是从它父亲的树推来的。

然后查询即可。

注意最后一行千万不能有换行,否则无限PE!

#include <cstdio>
#include <algorithm>
#define N 100001
using namespace std;
struct data
{
int num , rank;
}a[N];
int root[N] , lp[N << 5] , rp[N << 5] , sum[N << 5] , val[N] , top , tot;
int head[N] , to[N << 1] , next[N << 1] , cnt , fa[N] , bl[N] , deep[N] , si[N] , q[N] , tail;
bool cmp1(data a , data b)
{
return a.num < b.num;
}
bool cmp2(data a , data b)
{
return a.rank < b.rank;
}
void add(int x , int y)
{
to[++cnt] = y;
next[cnt] = head[x];
head[x] = cnt;
}
void dfs1(int x)
{
int i;
si[x] = 1;
for(i = head[x] ; i ; i = next[i])
{
if(to[i] != fa[x])
{
fa[to[i]] = x;
deep[to[i]] = deep[x] + 1;
dfs1(to[i]);
si[x] += si[to[i]];
}
}
}
void dfs2(int x , int c)
{
int i , k = 0;
bl[x] = c;
q[++tail] = x;
for(i = head[x] ; i ; i = next[i])
if(to[i] != fa[x] && si[to[i]] > si[k])
k = to[i];
if(k)
{
dfs2(k , c);
for(i = head[x] ; i ; i = next[i])
if(to[i] != fa[x] && to[i] != k)
dfs2(to[i] , to[i]);
}
}
int getlca(int x , int y)
{
while(bl[x] != bl[y])
{
if(deep[bl[x]] < deep[bl[y]])
swap(x , y);
x = fa[bl[x]];
}
if(deep[x] < deep[y]) return x;
return y;
}
void pushup(int x)
{
sum[x] = sum[lp[x]] + sum[rp[x]];
}
void ins(int x , int &y , int l , int r , int p)
{
y = ++tot;
if(l == r)
{
sum[y] = sum[x] + 1;
return;
}
int mid = (l + r) >> 1;
if(p <= mid) rp[y] = rp[x] , ins(lp[x] , lp[y] , l , mid , p);
else lp[y] = lp[x] , ins(rp[x] , rp[y] , mid + 1 , r , p);
pushup(y);
}
int query(int a , int b , int c , int d , int l , int r , int p)
{
if(l == r) return val[l];
int mid = (l + r) >> 1;
if(sum[lp[a]] + sum[lp[b]] - sum[lp[c]] - sum[lp[d]] >= p) return query(lp[a] , lp[b] , lp[c] , lp[d] , l , mid , p);
else return query(rp[a] , rp[b] , rp[c] , rp[d] , mid + 1 , r , p - sum[lp[a]] - sum[lp[b]] + sum[lp[c]] + sum[lp[d]]);
}
int main()
{
int n , m , i , x , y , z , f , last = 0;
scanf("%d%d" , &n , &m);
for(i = 1 ; i <= n ; i ++ )
{
scanf("%d" , &a[i].num);
a[i].rank = i;
}
sort(a + 1 , a + n + 1 , cmp1);
val[0] = 0x80000000;
for(i = 1 ; i <= n ; i ++ )
{
if(a[i].num != val[top]) val[++top] = a[i].num;
a[i].num = top;
}
sort(a + 1 , a + n + 1 , cmp2);
for(i = 1 ; i < n ; i ++ )
{
scanf("%d%d" , &x , &y);
add(x , y);
add(y , x);
}
dfs1(1);
dfs2(1 , 1);
for(i = 1 ; i <= tail ; i ++ )
ins(root[fa[q[i]]] , root[q[i]] , 1 , top , a[q[i]].num);
while(m -- )
{
scanf("%d%d%d" , &x , &y , &z);
x ^= last;
f = getlca(x , y);
last = query(root[x] , root[y] , root[f] , root[fa[f]] , 1 , top , z);
printf("%d" , last);
if(m) printf("\n");
}
return 0;
}

最新文章

  1. SQL-类型转换函数
  2. iOS 应用的生命周期
  3. 用EF访问Centos下的MySQL
  4. solr课程学习系列-solr服务器配置(2)
  5. list.clear()和list=null的区别
  6. spring源码学习之:spring容器的applicationContext启动过程
  7. linux的定时任务crontab
  8. Jordan Lecture Note-8: The Sequential Minimal Optimization Algorithm (SMO).
  9. ssh连接失败解决方法
  10. IndexedDB demo showcase
  11. eclipse 和myEclipse 项目导入
  12. word的标题行前面数字变成黑框 解决方案
  13. spring 4 升级踩雷指南
  14. EF时,数据库字段和实体类不一致问题
  15. NTFS权限笔记 2017-12-4
  16. Math java
  17. html中表单的应用
  18. MySQL 分组之后如何统计记录条数 gourp by 之后的 count()
  19. WinMain函数详解(转载再编辑)
  20. CentOS7 安装redis 并且设置成服务自动启动

热门文章

  1. HTML5 tricks for mobile
  2. 12 动态语言 __slots__
  3. JS学习 用 arguments 对象模拟函数重载
  4. C# 简单工厂
  5. adb获取设备的序列号
  6. 一段代码-Java
  7. 407. Plus One【LintCode java】
  8. Git版本库工作流程图想
  9. 预分配内存fifo实现可变长度字节序列存储
  10. ThinkPHP - 5 - 学习笔记(2015.4.15)