洛谷P2633/bzoj2588 Count on a tree (主席树)
洛谷P2633/bzoj2588 Count on a tree
题目描述
给定一棵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.
Solution
bzoj题面传送门
无修改,树上路径查询k小值
k小值可以主席树解决,树上路径其实不用真的提出来,我们可以运用差分的思路,由于是点差,求出它的\(lca,\)那么\(ans=a[u]+a[v]-a[lca]-a[fa[lca]]\).
具体到这道题就是\(u\)的主席树\(+v\)的主席树\(-lca\)的主席树\(-fa[lca]\)的主席树
\(lca\)倍增求就可以了
Code
#include<bits/stdc++.h>
#define in(i) (i=read())
#define il extern inline
#define rg register
#define mid ((l+r)>>1)
#define Min(a,b) ((a)<(b)?(a):(b))
#define Max(a,b) ((a)>(b)?(a):(b))
#define lol long long
using namespace std;
const int N=1e5+10;
int read() {
int ans=0, f=1; char i=getchar();
while (i<'0' || i>'9') {if(i=='-') f=-1; i=getchar();}
while (i>='0' && i<='9') ans=(ans<<1)+(ans<<3)+(i^48), i=getchar();
return ans*f;
}
int n,m,cur,num,tot,pre;
int head[N],nex[N<<1],to[N<<1];
int f[25][N],lg[N]={-1},dep[N];
int a[N],h[N],rt[N<<6];
struct Chair_Tree {
int l,r,v;
}t[N<<6];
void add(int a,int b) {
to[++cur]=b,nex[cur]=head[a],head[a]=cur;
to[++cur]=a,nex[cur]=head[b],head[b]=cur;
}
void insert(int &u,int l,int r,int pre,int p) {
t[u=++tot]=t[pre], t[u].v++;
if(l==r) return;
if(p<=mid) insert(t[u].l,l,mid,t[pre].l,p);
else insert(t[u].r,mid+1,r,t[pre].r,p);
}
int query(int a,int b,int c,int d,int l,int r,int k) {
int AQ=t[t[a].l].v+t[t[b].l].v-t[t[c].l].v-t[t[d].l].v;
if(l==r) return h[l];
if(k<=AQ) return query(t[a].l,t[b].l,t[c].l,t[d].l,l,mid,k);
else return query(t[a].r,t[b].r,t[c].r,t[d].r,mid+1,r,k-AQ);
}
void dfs(int u) {
a[u]=lower_bound(h+1,h+1+num,a[u])-h;
insert(rt[u],1,num,rt[f[0][u]],a[u]);
for (int i=head[u];i;i=nex[i]) {
if(to[i]==f[0][u]) continue;
dep[to[i]]=dep[u]+1, f[0][to[i]]=u;
dfs(to[i]);
}
}
void init() {
for (int i=1;i<=n;i++) lg[i]=lg[i>>1]+1;
for (int i=1;i<=lg[n];i++)
for (int j=1;j<=n;j++)
f[i][j]=f[i-1][f[i-1][j]];
}
int LCA(int a,int b) {
if(dep[a]>dep[b]) swap(a,b);
int s=dep[b]-dep[a];
for (int i=lg[s];i>=0;i--)
if(s>>i&1) b=f[i][b];
if(a==b) return a;
for (int i=lg[n];i>=0;i--) {
if(f[i][a]==f[i][b]) continue;
a=f[i][a], b=f[i][b];
}return f[0][a];
}
int main()
{
in(n), in(m);
for (int i=1;i<=n;i++) in(a[i]),h[i]=a[i];
sort(h+1,h+1+n); num=unique(h+1,h+1+n)-h-1;
for (int i=1,x,y;i<n;i++) in(x),in(y),add(x,y);
dep[1]=1, dfs(1), init();
for (int i=1,a,b,k;i<=m;i++) {
in(a),in(b),in(k),a^=pre;
int lca=LCA(a,b);
printf("%d\n",pre=query(rt[a],rt[b],rt[lca],rt[f[0][lca]],1,num,k));
}
}
洛谷P2633/bzoj2588 Count on a tree (主席树)的更多相关文章
- 【洛谷 P2633】 Count on a tree(主席树,树上差分)
题目链接 思维难度0 实现难度7 建出主席树后用两点的状态减去lca和lca父亲的状态,然后在新树上跑第\(k\)小 #include <cstdio> #include <cstr ...
- [bzoj2588][count on a tree] (主席树+lca)
Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始 ...
- BZOJ2588:Count on a tree(主席树)
Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始 ...
- 洛谷P2633 Count on a tree(主席树上树)
题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始为0,即第一个 ...
- 洛谷 P2633 Count on a tree 主席树
在一棵树上,我们要求点 $(u,v)$ 之间路径的第$k$大数. 对于点 $i$ ,建立 $i$ 到根节点的一棵前缀主席树. 简单容斥后不难得出结果为$sumv[u]+sumv[v]−sumv[l ...
- 【洛谷2633】Count on a tree(树上主席树)
点此看题面 大致题意: 给你一棵树,每次问你两点之间第\(k\)小的点权,强制在线. 主席树 这种题目强制在线一般就是数据结构了. 而看到区间第\(k\)小,很容易就能想到主席树. 至少不会有人想到树 ...
- 【BZOJ2588】Spoj 10628. Count on a tree 主席树+LCA
[BZOJ2588]Spoj 10628. Count on a tree Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lasta ...
- 【BZOJ-2588】Count on a tree 主席树 + 倍增
2588: Spoj 10628. Count on a tree Time Limit: 12 Sec Memory Limit: 128 MBSubmit: 3749 Solved: 873[ ...
- P2633 Count on a tree(主席树)
题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始为0,即第一个 ...
随机推荐
- Git + Gerrit 操作备忘
Git review 作用 可以用来提交代码审核到Gerrit 安装 使用pip 安装 git-review 插件,执行 sudo -H pip install git-review 使用示例 可以参 ...
- Amazon - removed your selling privileges and placed a temporary hold on any funds - 1
Hello, We are writing to let you know that we have removed your selling privileges and placed a temp ...
- ASP.NET Web API - 使用 Castle Windsor 依赖注入
示例代码 项目启动时,创建依赖注入容器 定义一静态容器 IWindsorContainer private static IWindsorContainer _container; 在 Applica ...
- django的htpp请求之WSGIRequest
WSGIRequest对象 Django在接收到http请求之后,会根据http请求携带的参数以及报文信息创建一个WSGIRequest对象,并且作为视图函数第一个参数传给视图函数.这个参数就是dja ...
- Java访问控制
转自:菜鸟教程
- 第五周PSP作业
PSP表格: 累积进度条: 折线图: 饼状图:
- Thunder——互评beta版本
基于NABCD和spec评论作品 Hello World!:http://www.cnblogs.com/vector121/p/7922989.html 欢迎来怼:http://www.cnblog ...
- YQCB项目介绍
YQCB记账本软件 制作人:YQCB团队 团队简介:团队成立于2017年11月21日,由陈美琪,张晨阳,邢全阳,刘昭为四人组成. 陈美琪:团队灵魂人物,背负着巨大的压力带起整个团队. 张晨阳:团队领军 ...
- FPGA论文
基于 NetFPGA 的 VCP 网络的设计与实现 --可变结构拥塞控制协议(VCP),适应于高带宽时延乘积网络的显式拥塞控制协议 无源光网络(PON) 1.区块链技术发展,物联网设备激增,服务器压力 ...
- 关《我是IT小小鸟》有感
我一直认为大学就是一个自由的舒适的学习环境,没有人可以干扰你限制你,以至于我到了大学之后只剩下了颓废的生活.每天上课玩手机,下课玩电脑,吃饭叫外卖,从不去锻炼,周末就熬夜通宵,状态越来越差,导致我逐渐 ...