静态主席树总结(静态区间的k大)

首先我们先来看一道题

给定N个正整数构成的序列,将对于指定的闭区间查询其区间内的第K小值。       输入格式:
第一行包含两个正整数N、M,分别表示序列的长度和查询的个数。
第二行包含N个正整数,表示这个序列各项的数字。
接下来M行每行包含三个整数 l, r, kl,r,k , 表示查询区间[l, r][l,r]         内的第k小值。
输出格式:
输出包含k行,每行1个正整数,依次表示每一次查询的结果

对于100%的数据满足:\(1 \leq N, M \leq 2\cdot 10^5\) \(1≤N,M≤2⋅10^5\)
对于数列中的所有数\(a_i\),都有\(-10^9\leq a_i\leq 10^9\)

基本思路

这个题目看上去很像一道线段树或者树状数组之类的裸题,但是仔细想想,区间第\(k\)小是线段树等数据结构维护不了的,这个时候,我们就需要引进一种新的数据结构,就是可持久化线段树,也就是主席树。(可持久化数据结构是可以访问历史版本的,这里也可以做到,但是这里不需要访问历史版本,我们只利用可持久化数据结构的思想)
主席树的本质上是N颗值域线段树(不知道值域线段树的请自行转走),对于每一颗线段树我们都维护从序列开始到这个元素的值域(即第\(i\)颗线段树维护的区间是第一号元素到第\(i\)号元素的值域)。
但实际上我们没有那么多时间和空间去维护\(n\)颗线段树,所以我们就要想,每一颗线段树由于值域相同,它们的形状是完全相同的,并且对于一颗第\(i\)颗线段树来说,它相对于第\(i-1\)颗线段树只增加了一个值,放在值域中,也就只有包含这个值的log个节点不同,所以对于每一颗线段树,我们只需要新开\(log\)个节点,其它的节点就用第\(i-1\)颗线段树的节点(如果你会可持久化数组,就会发现这其实跟可持久化数组很像,准确的说可持续化数组的模型就是主席树)。

注意要离散化
实现过程
---

构建

我们先开一个root数组来保存每一颗线段树的根,对于每一个线段树的节点记录它的值,左儿子和右儿子的编号,在构建第i颗线段树时,我们要同时访问第i-1颗线段树,每次构建一个节点之后,对于不包含这个新增值的儿子我们就直接将第i-1颗线段树的相应的那个儿子作为第i颗线段树的这个儿子。

比如说,假设离散化之后的值域是1~5,第i号元素是1,我们先构建根节点,然后发现这个节点左儿子的值域是1~2,右儿子的值域是3~5,右儿子的值域不包括1,所以右儿子就直接用第i-1颗线段树的右儿子,而此时我们就新建一个节点作为这个节点的左儿子,值为第i-1颗线段树的左儿子的值+1。

    int modify(int l,int r,int x,int k){//x表示上一颗线段树当前节点的标号//k表示需要新增的元素int y=++cnt;//新建当前节点,y位编号t[y]=t[x];//将上一颗线段树的节点的信息传递给当前节点t[y].x++;/*因为不包含k的节点不会被访问,所以实质上只要被访问过的节点都要加1*/if(l==r)return y;int mid=(l+r)>>1;if(k<=mid)t[y].l=modify(l,mid,t[x].l,k);else t[y].r=modify(mid+1,r,t[x].r,k);/*根据k值修改左右儿子信息*/return y;//将当前节点的编号号返回上一层}

查询

主席树的查询跟值域线段树的查询差不多,值域线段树的查询大家都会吧,我这里就不再赘述,不过,主席树每次需要同时查询两颗线段树,如果我们需要查询\([l,r]\)闭区间中第\(k\)小的值,我们就查询第\(l-1\)颗线段树和第\(r\)颗线段树的信息,由于所有线段树维护的值域完全一样,所以我们可以用第r颗线段树询问到的值减去第\(l-1\)颗线段树的值,就可以得出\([l,r]\)闭区间的值。(注意:你查询到的是离散之后的值,你需要输出的是离散之前的值)

具体实现过程

    int query(int l,int r,int la,int no,int k){//la,no,分别表示你要查询的两颗线段树的相应节点编号if(l==r)return l;/*如果节点内只有一个值,这就是第k大,直接返回*/int l1=t[la].l,l2=t[no].l,r1=t[la].r,r2=t[no].r;//l1,r1,l2,r2分别表示这两个节点的左右儿子。int s=t[l2].s-t[l1].s  ,  mid=(l+r)>>1;if(s>=k)return query(l,mid,l1,l2,k);else return query(mid+1,r,r1,r2,k-s);}

代码

    #include<bits/stdc++.h>using namespace std;inline int gi(){char a=getchar();int b=0;while(a<'0'||a>'9')a=getchar();while(a>='0'&&a<='9')b=b*10+a-'0',a=getchar();return b;}const int N=1e6+20;struct ljq{int x,id;}b[N];struct tree{int l,r,s;}t[N*5];int cmp(ljq x,ljq y){return x.x<y.x;}int a[N],p[N],root[N],n,m,cnt;void work1(){n=gi();m=gi();for(int i=1;i<=n;++i)b[i].x=gi(),b[i].id=i;sort(b+1,b+n+1,cmp);b[0].x=-2e9;for(int s=0,i=1;i<=n;++i){if(b[i].x!=b[i-1].x)p[++s]=b[i].x;a[b[i].id]=s;}}void bt(int l,int r,int x){if(l==r)return;int mid=(l+r)>>1;t[x].l=++cnt;t[x].r=++cnt;bt(l,mid,t[x].l);bt(mid+1,r,t[x].r);}void work2(int l,int r,int la,int no,int x){t[no].s=t[la].s+1;if(l==r)return;int mid=(l+r)>>1;t[no].l=t[la].l;t[no].r=t[la].r;if(x<=mid){t[no].l=++cnt;work2(l,mid,t[la].l,t[no].l,x);}else{t[no].r=++cnt;work2(mid+1,r,t[la].r,t[no].r,x);}}/*这个构建主席树的实现过程和上面略有不同,上面的更方便,是我在打带修改的主席树的时候写的,这里我懒得改了,仅做参考*/int query(int l,int r,int la,int no,int k){if(l==r)return l;int l1=t[la].l,l2=t[no].l,r1=t[la].r,r2=t[no].r;int s=t[l2].s-t[l1].s,mid=(l+r)>>1;if(s>=k)return query(l,mid,l1,l2,k);else return query(mid+1,r,r1,r2,k-s);}int main(){work1();root[0]=++cnt;bt(1,n,1);for(int i=1;i<=n;++i){root[i]=++cnt;work2(1,n,root[i-1],root[i],a[i]);}while(m--){int l=gi(),r=gi(),k=gi();int x=query(1,n,root[l-1],root[r],k);printf("%d\n",p[x]);}return 0;}

转载于:https://www.cnblogs.com/ljq-despair/p/8639345.html

静态主席树总结(静态区间的k大)相关推荐

  1. zoj 2112 树状数组 套主席树 动态求区间 第k个数

    总算是把动态求区间第k个数的算法看明白了. 在主席树的基础上,如果有修改操作,则要通过套树状数组来实现任意区间求第k小的问题. 刚开始看不明白什么意思,现在有一点理解.树状数组的每个元素是一个线段树, ...

  2. 主席树 ---- LCA(树上第k大)Count on a tree

    题目链接:https://www.spoj.com/problems/COT/en/ 题意: 给你一颗树,问你u,v结点这条路径上第k大是多少. 思路:就是说先将无根树转化为有根树,然后对每一条链建立 ...

  3. 【bzoj 十连测】[noip2016十连测第三场]Problem C: 序列(静态主席树)

    Problem C: [noip2016十连测第三场]序列 Time Limit: 10 Sec  Memory Limit: 256 MB Submit: 78  Solved: 32 [Submi ...

  4. hdu 5919--Sequence II(主席树--求区间不同数个数+区间第k大)

    题目链接 Problem Description Mr. Frog has an integer sequence of length n, which can be denoted as a1,a2 ...

  5. POJ 2104 K-th Number 主席树(区间第k大)

    题目链接: http://poj.org/problem?id=2104 K-th Number Time Limit: 20000MSMemory Limit: 65536K 问题描述 You ar ...

  6. J - [永不止步-2017]_区间第K大-线段树维护

    J - [永不止步-2017]_区间第K大 把线段树的结点的数据域设置为vector类型即可别的操作为区间更新模板 思路就是这样runtime error暂时没改对 #include<bits/ ...

  7. poj2104(区间第k大+离散化)

    题意: 给定一个序列,求[a,b]区间第k大的数字. 思路: 主席树模板题,但是注意数据范围,需要离散化. 代码: #include<cstdio> #include<cstring ...

  8. Dynamic Rankings——带修改区间第k大

    三种做法: 1.整体二分: 二分mid 考虑小于mid的修改的影响 但是大于mid的修改可能会干掉小于mid的一些值 所以额外把一个修改变成一个值的删除和一个值的添加 这样就相互独立了! 整体二分,树 ...

  9. O(n)复杂度求区间第K大

    O(n)复杂度求区间第K大 描述 给你一个数组,O(N)复杂度找出其中第K大的数. 输入 第一行包括两个数,N,K,分别代表数组大小,以及你应该找出第K大的数. 接下来N行,每行包括一个数. ​ 输出 ...

最新文章

  1. java cometd_关于cometd的一些经验总结-java端
  2. leecode第一百四十八题(排序链表)
  3. .NET CORE(C#) WPF 值得推荐的动画菜单设计
  4. 【转载】浅析输入法原理
  5. 2020年最新的过某宝滑块验证技术,Python大牛轻松搞定技术难题
  6. 各种initcall的执行先后顺序(module_init、postcore_initcall、arch_initcall、subsys_initcall、 fs_initcall)【转】...
  7. [译] 如何用 Python 写一个 Discord 机器人
  8. STM32标准库与HAL库中的Mode和Pull设置
  9. 软件设计原则(三) 依赖倒置原则
  10. 《Go Web编程实战派——从入门到精通》学习笔记之第1章 Go基础入门
  11. Windows安装杜比音效驱动
  12. 五胡十六国、东晋南北朝这280年历史,你知道多少?5000字带你看个清楚明白
  13. 怎样在Apple Silicon M1 Mac上引导到恢复模式
  14. HTML页面基本结构
  15. linux 命令:du 详解
  16. python撩妹技能_干货必看 | 手把手教你用Python撩妹
  17. 搭建 Nexus 私服
  18. 使用GraalVM实现java调用python脚本
  19. 腾讯安全【数实融合 安全共赢】圆桌栏目,精彩来袭
  20. KEGG pathway 数据库

热门文章

  1. git 32位_编译64位的BorderlessGaming
  2. python 函数递归_Python零基础之三元表达式、函数递归、匿名函数教程!超级详细!...
  3. 服务器用户配置文件在哪里找,管理远程桌面服务的用户配置文件
  4. oracle中delete、truncate、drop的区别 (转载)
  5. Node.js Up and Runing 学习日记(八)
  6. Python多线程3:queue
  7. Xtreme8.0 - Kabloom dp
  8. openfire学习4---android客户端聊天开发之聊天功能开发
  9. python selenium自动化(三)Chrome Webdriver的兼容
  10. 好消息:VS 2008 and .NET 3.5 Beta 2 发布了