题目描述

给定一棵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行,表示每个询问的答案。

输入输出样例

输入样例#1: 复制

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

输出样例#1: 复制

2

8

9

105

7

说明

HINT:

N,M<=100000

暴力自重。。。

来源:bzoj2588 Spoj10628.


题解

一道比较涨见识的题目吧。

在这里我们要统计一条路径上的Kth值,考虑把路径转化为一条序列。怎么转换?

我们可以让新增的那个点在原有的fa节点上再建树。这样我们就可以利用树上差分了。

但是我们会发现,x,y两个节点减去两次它们共同的LCA时,LCA的值就没有算在里面了,这个时候我们就不要减去两次LCA应该减去一次LCA,减去一次fa[LCA]。这里的点都是指在主席树相对应的状态。

代码

#include<cstdio>
#include<cstring>
#include<iostream>
#include<cmath>
#include<algorithm>
using namespace std;
const int N=100001;
int top[N],size[N],fa[N],dep[N],son[N];
int ch[N],n,m,q,b[N],tr[N<<5],sum[N<<5],tot,l[N<<5],r[N<<5];
int num,head[N],cnt;
struct node{
int to,next;
}e[N<<1];
int read()
{
int x=0,w=1;char ch=getchar();
while(ch>'9'||ch<'0'){if(ch=='-')w=-1;ch=getchar();}
while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
return x*w;
} void add(int from,int to){
num++;
e[num].to=to;
e[num].next=head[from];
head[from]=num;
} void dfs1(int x){
size[x]=1;
for(int i=head[x];i;i=e[i].next){
int v=e[i].to;
if(!dep[v]){
dep[v]=dep[x]+1;fa[v]=x;
dfs1(v);size[x]+=size[v];
if(son[son[x]]<size[v])son[x]=v;
}
}
} void dfs2(int x,int tp){
top[x]=tp;if(son[x])dfs2(son[x],tp);
for(int i=head[x];i;i=e[i].next){
int v=e[i].to;if(v!=fa[x]&&v!=son[x])dfs2(v,v);
}
} int cal(int x,int y){
int fx=top[x],fy=top[y];
while(fx!=fy){
if(dep[fx]<dep[fy])swap(fx,fy),swap(x,y);
x=fa[fx],fx=top[x];
}
if(dep[x]>=dep[y])return y;return x;
} int build(int left,int right){
int mid=(left+right)>>1;
int root=++cnt;
sum[root]=0;
if(left<right){
l[root]=build(left,mid);
r[root]=build(mid+1,right);
}
return root;
} int update(int pre,int left,int right,int v){
int mid=(left+right)>>1;
int root=++cnt;
l[root]=l[pre];r[root]=r[pre];sum[root]=sum[pre]+1;
if(left<right){
if(v<=mid)l[root]=update(l[pre],left,mid,v);
else r[root]=update(r[pre],mid+1,right,v);
}
return root;
} int query(int u1,int u2,int v1,int v2,int left,int right,int k){
int mid=(left+right)>>1;
int x=sum[l[u1]]+sum[l[u2]]-sum[l[v1]]-sum[l[v2]];
if(left>=right)return left;
if(x>=k)return query(l[u1],l[u2],l[v1],l[v2],left,mid,k);
else return query(r[u1],r[u2],r[v1],r[v2],mid+1,right,k-x);
} void dfs(int x){
int t=lower_bound(b+1,b+m+1,ch[x])-b;
tr[x]=update(tr[fa[x]],1,m,t);
for(int i=head[x];i;i=e[i].next){
int v=e[i].to;
if(v!=fa[x])dfs(v);
}
} int main()
{
n=read();q=read();
for(int i=1;i<=n;i++)
{
b[i]=ch[i]=read();
}
sort(b+1,b+n+1);
m=unique(b+1,b+n+1)-b-1;
for(int i=1;i<n;i++)
{
int x=read(),y=read();
add(x,y);add(y,x);
}
fa[1]=0;dep[1]=1;
dfs1(1);dfs2(1,0);
tr[0]=build(1,m);
dfs(1);
int last=0;
while(q--)
{
int l=read(),r=read(),k=read();
int lca=cal(l^last,r);
int ff=fa[lca];
printf("%d\n",last=b[query(tr[l^last],tr[r],tr[lca],tr[ff],1,m,k)]);
}
return 0;
}

P2633 Count on a tree(主席树)的更多相关文章

  1. 洛谷P2633 Count on a tree(主席树上树)

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

  2. 洛谷 P2633 Count on a tree 主席树

    在一棵树上,我们要求点 $(u,v)$ 之间路径的第$k$大数. 对于点 $i$  ,建立 $i$  到根节点的一棵前缀主席树. 简单容斥后不难得出结果为$sumv[u]+sumv[v]−sumv[l ...

  3. 洛谷P2633 Count on a tree 主席树

    传送门:主席树 解题报告: 传送门! umm这题我还麻油开始做 所以 先瞎扯一波我的想法,如果错了我就当反面教材解释这种典型错误,对了我就不管了QwQ 就直接dfs,在dfs的过程中建树 然后就直接查 ...

  4. 洛谷P2633/bzoj2588 Count on a tree (主席树)

    洛谷P2633/bzoj2588 Count on a tree 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K ...

  5. 【BZOJ-2588】Count on a tree 主席树 + 倍增

    2588: Spoj 10628. Count on a tree Time Limit: 12 Sec  Memory Limit: 128 MBSubmit: 3749  Solved: 873[ ...

  6. spoj cot: Count on a tree 主席树

    10628. Count on a tree Problem code: COT You are given a tree with N nodes.The tree nodes are number ...

  7. Bzoj 2588: Spoj 10628. Count on a tree 主席树,离散化,可持久,倍增LCA

    题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2588 2588: Spoj 10628. Count on a tree Time Limit ...

  8. 【BZOJ2588】Spoj 10628. Count on a tree 主席树+LCA

    [BZOJ2588]Spoj 10628. Count on a tree Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lasta ...

  9. SPOJ Count on a tree(主席树+LCA)

    一.题目 COT - Count on a tree You are given a tree with N nodes. The tree nodes are numbered from 1 to  ...

随机推荐

  1. HDU 5234 Happy birthday【DP】

    题意:给出n*m的格子,每个格子的值为w[i][j],在值不超过k的时候,可以往右或者往下走,问从(1,1)走到(n,m)能够得到的最大的值 类似于背包 d[i][j][k]=maxx(d[i-1][ ...

  2. php截取字符串|php截取字符串前几位|php截取中文字符串

    转 截取字符串专题:php截取字符串函数,php 字符串长度,php截取字符串前几位 PHP截取中文字符串(mb_substr)和获取中文 => http://www.q3060.com/lis ...

  3. Hadoop_HDFS-基础知识摘要

    Hadoop典型应用有:搜索.日志处理.推荐系统.数据分析.视频图像分析.数据保存等.0.数据要首先分块 Block:将一个文件进行分块,通常是64M. NameNode:--管理节点保存整个文件系统 ...

  4. 利用MFC创建窗口、消息映射、window中的字节

    利用MFC创建窗口: 1.mfc的头文件:afxwin.h 2.自定义类,继承于CWinApp,应用程序类(app应用程序对象,有且仅有一个) 3.程序入口:Initinstance 4.在程序入口中 ...

  5. vue-quill-editor-upload : 实现vue-quill-editor上传图片到服务器

    vue-quill-editor-upload git: https://github.com/NextBoy/vu... A plug-in for uploading images to your ...

  6. CloudStack云基础架构的一些概念

    1. Zones(区域) 一个区域在CloudStack配置中是最大的组织单元.一个区域通常代表一个单独的数据中心,虽然在一个数据中心也允许有多个区域.将基础架构设施加入到区域中的好处是提供物理隔离和 ...

  7. 如何检查 Android 应用的内存使用情况

    Android是为移动设备而设计的,所以应该关注应用的内存使用情况.尽管Android的Dalvik虚拟机会定期执行垃圾回收操作,但这也不意味着就可以忽视应用在何时何处进行内存分配和释放.为了提供良好 ...

  8. HDU--4891--The Great Pan--暴力搜索

    The Great Pan Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) To ...

  9. POJ 2828 Buy Tickets(线段树&#183;插队)

    题意  n个人排队  每一个人都有个属性值  依次输入n个pos[i]  val[i]  表示第i个人直接插到当前第pos[i]个人后面  他的属性值为val[i]  要求最后依次输出队中各个人的属性 ...

  10. 51nod-1189: 阶乘分数

    [传送门:51nod-1189] 简要题意: 给出一个数n,求出有多少个正整数x,y(0<x<=y)满足$1/n!=1/x+1/y$ 题解: 一开始还以为不可做 结果推一下柿子就会了 $1 ...