主席树总结(经典区间第k小问题)(主席树,线段树)
接着上一篇总结——可持久化线段树来整理吧。点击进入
这两种数据结构确实有异曲同工之妙。结构是很相似的,但维护的主要内容并不相同,主席树的离散化、前缀和等思想也要更难理解一些。
闲话
话说刚学习主席树的时候百度了一下,看到了“主席树”这一名字的由来——
线段树竟然是被一个黄嘉泰的大佬因不会划分树来代替的,,,,,因缩写是HJT取名为主席树= =!orz
我的内心瞬间感到了不安。。。。。。(看我的名字,就在右边)
能跟神犇有相同的缩写是何等荣幸!看来这个主席树我得好好学了。
那么接下来进入正题。
主席树
直接从最经典的应用——区间第\(k\)小问题开始吧。ZSY巨佬对这一问题的思路挺清晰的,本蒟蒻在这里也参考一下。%ZSY%请点这里
1. 静态区间第k小问题
洛谷题目传送门
即给出一个序列,每次询问求给定区间\([l,r]\)内第\(k\)小的值
思路分析
先不考虑每一个区间的情况,从最简单的查询整个区间\([1,r]\)的情况开始。
对数据离散化后,用一个线段树来维护,每个节点维护对应离散化后值区间的数的总个数\(size\)。自上至下进行询问操作时,判断当前点左子树的\(size\)与要查询的排名\(k\)的大小关系。如果小于等于,就到左子树中找,\(k\)不变。否则到右子树中找排名\(k-size\)的值。这与平衡树(Splay,Treap)等查询给定排名数的方法是基本一样的。
那对于所有可能的区间,又该怎样维护呢?我们其实只要\(N\)个线段树就好了,第\(i\)个线段树维护\([1,i]\)的情况。这里我们利用了前缀和的性质。查询\([l,r]\)就等于查询\([1,r]\)减去\([1,l-1]\)的对应的\(size\)没错吧。因为线段树是完全二叉树,具有结构稳定的性质,所以\(N\)个线段树长得是一样的,对应区间相减是可行的。
然而暴力开\(N\)个空间保准炸掉。这时候我们回头看看可持久化线段树是怎么做的。没错,从\([1,i-1]\)到\([1,i]\)也只变了一个值!于是同样只要新开\(log\)个节点,保存\(root_i\)到\(i\)对应叶子节点的路径就OK了。
拿洛谷题目里的样例来几张图吧,好理解些。
首先是离散化后的序列。

我们一开始要建一棵空线段树,除了有个结构,所有的\(size\)均为\(0\)。然后一个一个的添加线段树。加入\([1,1]\)的线段树后会是这样:

再加入\([1,2]\):

后面的手推一下吧。。。。。。
至此,\(N\)棵树就建好了,并且只用了\(N \log N\)的空间。
查询的时候存两个点,一开始为\(root_r\)和\(root_{l-1}\),两个\(size\)的差与\(k\)来比大小,两个点要同时往左/右跳。
更多细节就看代码吧。
#include<cstdio>
#include<algorithm>
using namespace std;
#define R register int
const int N=200009,M=5000009;
int P,a[N],b[N],rt[N],lc[M],rc[M],s[M];
#define G c=getchar()
inline void in(R&z)
{
register bool f=0;
register char G;
while(c<'-')G;
if(c=='-')f=1,G;
z=c&15;G;
while(c>'-')z=(z<<3)+(z<<1)+(c&15),G;
if(f)z=-z;
}//快读
void build(R&t,R l,R r)
{
t=++P;
if(l!=r)
{
R m=(l+r)>>1;
build(lc[t],l,m);
build(rc[t],m+1,r);
}
}//线段树操作,建一个空线段树
inline void insert(R*t,R u,R l,R r,R v)
{
while(l!=r)
{
s[*t=++P]=s[u]+1;//注意这里要+1
R m=(l+r)>>1;
if(v<=m)r=m,rc[*t]=rc[u],t=&lc[*t],u=lc[u];
else l=m+1,lc[*t]=lc[u],t=&rc[*t],u=rc[u];
}
s[*t=++P]=s[u]+1;
}//插入操作在可持久化线段树总结里面有更详细的介绍
inline int ask(R t,R u,R l,R r,R k)
{
while(l!=r)
{
R m=(l+r)>>1,v=s[lc[u]]-s[lc[t]];//作差
if(k<=v)r=m,t=lc[t],u=lc[u];//两个点一起跳
else l=m+1,t=rc[t],u=rc[u],k-=v;
}
return b[l];
}
int main()
{
R n,m,i,l,r,sz;
in(n);in(m);
for(i=1;i<=n;++i)
in(a[i]),b[i]=a[i];
sort(b+1,b+n+1);
sz=unique(b+1,b+n+1)-b-1;//离散化,排序去重
build(rt[0],1,sz);
for(i=1;i<=n;++i)
insert(&rt[i],rt[i-1],1,sz,lower_bound(b+1,b+sz+1,a[i])-b);//直接用STL的二分找对应值了
while(m--)
{
in(l);in(r);in(i);
printf("%d\n",ask(rt[l-1],rt[r],1,sz,i));
}
return 0;
}
2. 动态区间第k小问题
洛谷题目传送门
就是比上面那题多了个修改。。。。。。
改为树状数组维护前缀和,使修改复杂度降低。
详细题解在此
3.树上路径第k小问题
洛谷题目传送门
只维护树根节点(随便设)到每个节点的前缀和,查询时\(size[root,u]+size[root,v]-size[root,lca(u,v)]-size[root,father(lca[u,v))]\)即为\(u->v\)路径的\(size\)。
于是用倍增求LCA。
详细题解在此
主席树总结(经典区间第k小问题)(主席树,线段树)的更多相关文章
- HDU 2665.Kth number-可持久化线段树(无修改区间第K小)模板 (POJ 2104.K-th Number 、洛谷 P3834 【模板】可持久化线段树 1(主席树)只是输入格式不一样,其他几乎都一样的)
Kth number Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- Dynamic Rankings || 动态/静态区间第k小(主席树)
JYF大佬说,一星期要写很多篇博客才会有人看 但是我做题没有那么快啊QwQ Part1 写在前面 区间第K小问题一直是主席树经典题=w=今天的重点是动态区间第K小问题.静态问题要求查询一个区间内的第k ...
- ZOJ 2112 Dynamic Rankings(树状数组套主席树 可修改区间第k小)题解
题意:求区间第k小,节点可修改 思路:如果直接用静态第k小去做,显然我更改一个节点后,后面的树都要改,这个复杂度太高.那么我们想到树状数组思路,树状数组是求前缀和,那么我们可以用树状数组套主席树,求出 ...
- SPOJ-COT-Count on a tree(树上路径第K小,可持久化线段树)
题意: 求树上A,B两点路径上第K小的数 分析: 同样是可持久化线段树,只是这一次我们用它来维护树上的信息. 我们之前已经知道,可持久化线段树实际上是维护的一个前缀和,而前缀和不一定要出现在一个线性表 ...
- [luogu3834]静态区间第k小【主席树】
传送门:https://www.luogu.org/problemnew/show/P3834 题目描述 如题,给定N个整数构成的序列,将对于指定的闭区间查询其区间内的第K小值. 分析 很多人都说是用 ...
- 主席树铺垫——总区间第k小
题目描述(口糊) 先给定一个长度为n的数列,然后给m次操作,每次输入b,求第b小的数. 样例输入 5 7 4 10 9 23 5 1 2 3 4 5 样例输出 4 7 9 10 23 数据范围及温馨提 ...
- 主席树(可持久化线段树)静态区间第K小
传送门主席树 #include <bits/stdc++.h> #define int long long using namespace std; const int maxn=2e5+ ...
- poj2104 K-th Number区间第k小值 主席树
原来主席树就是可持久化线段树啊,刚知道,,, 作为一道裸题,还是必A的,然而一开始偷懒不写离散化跪了N多遍,后来在缪大的帮助下发现了这个问题,遂A之 ——又是这种破问题,实在不想说自己了 把n个数看成 ...
- BZOJ 3065 带插入区间K小值(sag套线段树)
3065: 带插入区间K小值 Time Limit: 60 Sec Memory Limit: 512 MBSubmit: 4696 Solved: 1527[Submit][Status][Di ...
随机推荐
- main函数的实现解析
main函数的传参的实现,其实也是一个解析字符串的过程:将每个word后一个空格改为“/0”,将单词提取出来. 就是这么简单. 废话不多说,直接上代码: #include<stdio.h> ...
- yii2 源码分析Action类分析 (六)
Action类是控制器的基类, <?php namespace yii\base; use Yii; /** * Action是所有控制器动作类的基类,它继承组件类 * * 动作提供了重用动作方 ...
- jumpserver在centos 7上的部署
cd /opt/git clone https://github.com/jumpserver/jumpserver.gitcd jumpservergit checkout master 准备安装: ...
- windows下apache服务器开启压缩和网页缓存
找到配置文件:http.conf apache开启压缩 一.开启配置,去除下面代码前面的#号LoadModule deflate_module modules/mod_deflate.soLoadMo ...
- Chrome Stylist 插件 (CSS备份)
Stylist 插件还是很好用的,可以给网站自定义CSS样式,(还有个插件叫"油猴子",可以给网页加载自定义JS): 不过麻烦的是,现在的最新版360浏览器不能显示这个插件(这个浏 ...
- Java中InputStream装饰器模式的大家族
本文写在po主初学JAVA时,在学习inputStream摸不着头脑,受Java IO-InputStream家族 -装饰者模式一文启发,所以在理清思路时写下本文.因为初学,如有错误,望指正. 因为和 ...
- java 23种设计模式 深入理解
以下是学习过程中查询的资料,别人总结的资料,比较容易理解(站在各位巨人的肩膀上,望博主勿究) 创建型抽象工厂模式 http://www.cnblogs.com/java-my-life/archive ...
- asp.net core 一 Centos 环境部署
.netcore的运行环境,创建asp.net core 项目 CentOS 7 ,dotnet-sdk-2.0.0-2.0.0-1.x86_64 直接在liunx创建项目并运 ...
- MysqL错误之_ERROR! MySQL server PID file could not be found!
在配置Mysql主从GTID模式下,启动Mysql服务时出现报错,搜索了一番,找到了一个简单可靠的方法,直接成功.如果遇到相同问题没有解决的童鞋,那就去试一下很多其他方案,如,强制杀掉进程重启,修改其 ...
- 10分钟入门spark
Spark是硅谷各大公司都在使用的当红炸子鸡,而且有愈来愈热的趋势,所以大家很有必要了解学习这门技术.本文其实是笔者深入浅出hadoop系列的第三篇,标题里把hadoop去掉了因为spark可以不依赖 ...