洛谷题目传送门

题目大意

就是给你一棵树,每个点都有点权,每次任意询问两点间路径上点权第k小的值(强制在线)。

思路分析

第k小。。。。。。又是主席树了。但这次变成树了,无法直接维护前缀和。

又是树上差分的小套路——每一个点到根的前缀和还是很好维护对吧。

询问\(u,v\)的时候,我们可以知道\(size[root,u]\)和\(size[root,v]\)的和。

但我们需要的只是一条路径,\(lca(u,v)\)以上的全不要,\(lca(u,v)\)也只要算一次。

于是用\(size[root,u]+size[root,v]-size[root,lca(u,v)]-size[root,father(lca(u,v))]\),也就是询问的时候四个点一起跳。

求LCA最方便的是倍增法(不会的百度一下),还有每个点对应的线段树从其父亲的线段树继承而来(根节点从\(0\)号空线段树继承而来),这两个操作我们在一次dfs建树时就可以一并处理完了。

话说我好久没打过倍增LCA了,老是写挂。。。。。。

闲话

另外,本蒟蒻听说倍增的二维数组把长度小的那一维度(即表示\(2^j\)的维度)开在前面会跑的快一些。

于是本蒟蒻亲自验证了一下,开在前面1112ms,开在后面992ms(没开O2,开了以后两个差不多)。。。。。。

这又是什么鬼?!

难道有评测机的不稳定因素?(洛谷一直很稳定啊!)

或者是有其他原因?欢迎各位大佬指教。

下面贴代码(开在前面):

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define R register int
#define G c=getchar()
#define in(Z) G;\
while(c<'-')G;\
Z=c&15;G;\
while(c>'-')Z*=10,Z+=c&15,G;
const int N=100009,M=4000009;
int P,he[N],ne[N<<1],to[N<<1],f[18][N],d[N],b[N];
int n,SZ,a[N],g[N],rt[N],lc[M],rc[M],s[M];
void build(R&u,R l,R r){//建初始空线段树
u=++P;
if(l!=r){
R m=(l+r)>>1;
build(lc[u],l,m);
build(rc[u],m+1,r);
}
}
inline void insert(R*u,R v,R k){//更新
R l=1,r=SZ,m;
while(l!=r){
s[*u=++P]=s[v]+1;
m=(l+r)>>1;
if(k<=m)r=m,rc[*u]=rc[v],u=&lc[*u],v=lc[v];
else l=m+1,lc[*u]=lc[v],u=&rc[*u],v=rc[v];
}
s[*u=++P]=s[v]+1;
}
inline int ask(R u,R v,R w,R x,R k){//询问,四个点一起搞
R l=1,r=SZ,m,q;
while(l!=r){
m=(l+r)>>1;
q=s[lc[u]]+s[lc[v]]-s[lc[w]]-s[lc[x]];
if(k<=q)r=m,u=lc[u],v=lc[v],w=lc[w],x=lc[x];
else l=m+1,u=rc[u],v=rc[v],w=rc[w],x=rc[x],k-=q;
}
return g[l];
}
void dfs(R u,R fa){//建好树,顺便预处理LCA
d[u]=d[fa]+1;
f[0][u]=fa;
for(R&i=b[u];f[i+1][u]=f[i][f[i][u]];++i);
insert(&rt[u],rt[fa],lower_bound(g+1,g+SZ+1,a[u])-g);
for(R i=he[u];i;i=ne[i])
if(to[i]!=fa)dfs(to[i],u);
}
inline int getlca(R u,R v){//查LCA
R i;
if(d[u]<d[v])i=u,u=v,v=i;
for(i=b[u];d[u]>d[v]&&i>=0;--i)
if(d[f[i][u]]>=d[v])u=f[i][u];
for(i=b[u];i>=0;--i)
if(f[i][u]!=f[i][v])u=f[i][u],v=f[i][v];
return u==v?u:f[0][u];//这里老是忘记判
}
int main(){
register char c;
R p=1,m,i,u,v,k,lca,lans=0;
in(n);in(m);
for(i=1;i<=n;++i){in(a[i]);}
memcpy(g,a,(n+1)<<2);//搞出来离散化
sort(g+1,g+n+1);
SZ=unique(g+1,g+n+1)-g-1;
build(rt[0],1,SZ);
for(i=1;i<n;++i)
{
in(u);in(v);
to[++p]=v;ne[p]=he[u];he[u]=p;
to[++p]=u;ne[p]=he[v];he[v]=p;//建边
}
dfs(1,0);
while(m--){
in(u);in(v);in(k);
lca=getlca(u^=lans,v);
printf("%d\n",lans=ask(rt[u],rt[v],rt[lca],rt[f[0][lca]],k));
}
return 0;
}

洛谷P2633 Count on a tree(主席树,倍增LCA,树上差分)的更多相关文章

  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 Count on a tree

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

  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. 洛谷P2633 Count on a tree(主席树,倍增LCA)

    洛谷题目传送门 题目大意 就是给你一棵树,每个点都有点权,每次任意询问两点间路径上点权第k小的值(强制在线). 思路分析 第k小......又是主席树了.但这次变成树了,无法直接维护前缀和. 又是树上 ...

  7. ☆ [洛谷P2633] Count on a tree 「树上主席树」

    题目类型:主席树+\(LCA\) 传送门:>Here< 题意:给出一棵树.每个节点有点权.问某一条路径上排名第\(K\)小的点权是多少 解题思路 类似区间第\(K\)小,但放在了树上. 考 ...

  8. 洛谷P2633 Count on a tree

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

  9. 洛谷 P2633 Count on a tree 题解

    题面 对于每个点建立一颗主席树: 然后按照树上差分的思想统计主席树的前缀和: lca+主席树+前向星存表就可以了: #include <bits/stdc++.h> #define inc ...

随机推荐

  1. 20155233 《网络对抗》Exp4 恶意代码分析

    使用schtasks指令监控系统运行 先在C盘目录下建立一个netstatlog.bat文件,用来将记录的联网结果格式化输出到netstatlog.txt文件中,netstatlog.bat内容为: ...

  2. tensorflow batch

    这两天一直在看tensorflow中的读取数据的队列,说实话,真的是很难懂.也可能我之前没这方面的经验吧,最早我都使用的theano,什么都是自己写.经过这两天的文档以及相关资料,并且请教了国内的师弟 ...

  3. POJ3267

    从今天开始POJ里的一部分类型的题目就一般不放在一起写了 一个是太丑,格式麻烦,第二个是以后的题目难度都有所增大,因此一道题可能就要写蛮长 尤其是DP这一块,以前一直没好好学习,现在从基础的先开始吧 ...

  4. Hadoop日记Day5---HDFS介绍

    一.HDFS介绍 1.1 背景 随着数据量越来越大,在一个操作系统管辖的范围存不下了,那么就分配到更多的操作系统管理的磁盘中,但是不方便管理和维护,迫切需要一种系统来管理多台机器上的文件,这就是分布式 ...

  5. 【第十一课】Tomcat原理解析【转】

    目录 一.Tomcat顶层架构 二.Tomcat顶层架构小结: 三.Connector和Container的微妙关系 四.Connector架构分析 五.Container架构分析 六.Contain ...

  6. 5、Docker网络配置(单机)

    一.概述 以下内容参考:https://docs.docker.com/network/#network-drivers Docker容器和服务如此强大的原因之一是您可以将它们连接在一起,或者将它们连 ...

  7. HTML 图像实例

    61.插入图像本例演示如何在网页中显示图像.图像标签(<img>)和源属性(Src)在 HTML 中,图像由 <img> 标签定义. <img> 是空标签,意思是说 ...

  8. 前端菜鸟起飞之学会ps切图

    由于之前只顾着追求效率,没有学习过PS,但其实这是前端开发人员需要学会的技能之一,曾经看过一个大佬的前端经验分享说他在招聘时遇到不会切图的会直接pass掉,可见前端开发人员学会切图是多么重要.通过观看 ...

  9. centos crontab 计划任务 设置与查看

    centos 上 crontab 计划任务 ,这个版本解释的比较清晰 林涛 发表于:2017-4-27 11:11 分类:26点 标签:crontab,Linux,计划任务 36次 这个版本的cron ...

  10. linux第十八章学习笔记

    第十八章 调试 内核级开发的调试工作远比用户级开发艰难,它带来的风险比用户级别更高. 一.准备开始 1. 准备工作需要: 一个bug 一个藏匿bug的内核版本 相关内核代码的知识和运气 2. 在用户级 ...