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

静态主席树总结(静态区间的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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/255114.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java中MySQL事务处理举例

实例&#xff08;以sql语句中的insert语句为例&#xff09; import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement;/*** 事务的基本用法**/ …

永磁交流伺服电机的工作原理与更换新编码器后的常规零位校正方法

http://wuhuotun.blog.163.com/blog/static/73085450200910655748516/ 永磁交流伺服电机的编码器相位为何要与转子磁极相位对齐 其唯一目的就是要达成矢量控制的目标&#xff0c;使d轴励磁分量和q轴出力分量解耦&#xff0c;令永磁交流伺服电机定子绕组产生的电磁场始终正交于…

理解Java中字符流与字节流的区别

1. 什么是流 Java中的流是对字节序列的抽象&#xff0c;我们可以想象有一个水管&#xff0c;只不过现在流动在水管中的不再是水&#xff0c;而是字节序列。和水流一样&#xff0c;Java中的流也具有一个“流动的方向”&#xff0c;通常可以从中读入一个字节序列的对象被称为输入…

HALCON示例程序inspect_solar_fingers.hdev太阳能电池板电路缺陷检测

HALCON示例程序inspect_solar_fingers.hdev太阳能电池板电路缺陷检测 示例程序源码&#xff08;加注释&#xff09; 关于显示类函数解释 dev_update_off () dev_close_window () ImageName : ‘solar_cell/solar_cell_’ read_image (Image, ImageName ‘01’) dev_open_win…

C++多线程 例子

C多线程 例子2008-08-21 15:11//这是2个线程模拟卖火车票的小程序#include <windows.h>#include <iostream.h>DWORD WINAPI Fun1Proc(LPVOID lpParameter);//thread dataDWORD WINAPI Fun2Proc(LPVOID lpParameter);//thread dataintindex0;inttickets10;HANDLE hM…

2018/03/25

2019独角兽企业重金招聘Python工程师标准>>> March 25 2018 Sunday Weather&#xff1a;cloudy 1、需求&#xff1a; a0.5 b3 ca*b 求c的值&#xff1a; [rootDasoncheng sbin]# cat a.sh #!/bin/bash a0.5 b3 cecho $a*$b |bc echo $canswer referred&#xff1a;…

统计字符串中每种字符类型的个数demo

/** 统计字符串中每中字符类型的个数&#xff0c;思路&#xff1a;* 1、键盘录入一个字符串* 2、获取到每个字符&#xff0c;遍历字符串* for (int i 0; i < s.length(); i) {char ch s.charAt(i);* 3、判断每个字符属于哪种类型&#xff0c;对应的个数*/package cn.strin…

css3 图片放大缩小闪烁效果

直接把图片替换就可以了&#xff0c;我的图片是透明的&#xff0c;所以body设置为黑色的&#xff0c;不不要可以去掉 <!doctype html><html lang"en"><head> <meta charset"UTF-8"> <title>css3 图片放大缩小闪烁效果</t…

HALCON示例程序max_connection.hdev确定分割区域的最大区域数目

HALCON示例程序max_connection.hdev确定分割区域的最大区域数目 示例程序源码&#xff08;加注释&#xff09; read_image (Image, ‘monkey’) get_system (‘max_connection’, Information) set_system (‘max_connection’, 0) threshold (Image, Region, 128, 255) 区域…

elasticsearch分词聚合查询demo

2019独角兽企业重金招聘Python工程师标准>>> 我们在通过elasticsearch查询text类型的字段时&#xff0c;我们使用aggs进行聚合某个text类型field。这时elasticsearch会自动进行分词将分词后的结果进行聚合。获取每一个分词出现在文档的文档个数。注意&#xff1a;是…

软件工程进度条-第十五周

第十五周 所花时间&#xff08;包括上课&#xff09; 23 代码量&#xff08;行&#xff09; 1200 博客量&#xff08;篇&#xff09; 6 了解到的知识点 1、了解ListView的基本用法&#xff0c;并改变焦点触碰事件&#xff1b; 2、理解团队开发后进行软件项目总结的益处…

Spring实战第七章

一、SpringMVC配置代替方案 1自定DispatcherServlet 按照AbstractAnnotationConfigDispatcherServletInitializer的定义&#xff0c;它会创建DispatcherServlet和ContextLoaderListener。 AbstractAnnotationConfigDispatcherServletInitializer有三个方法是必须要重载的abstra…

EPSON TCP/IP 通信

EPSON SCARA机器人TCP/IP 通信时&#xff0c;涉及到的相关指令说明。 14.3 TCP/IP命令。 OpenNet //打开TCP/IP端口。 ChkNet //返回端口状态&#xff1a;等待读取的字节数或错误条件。 CloseNet //关闭TCP/IP端口。 SetNet //运行时或从命令窗口中设置通信端…

JDBC(九)DatabaseMetaData 数据库元数据

通过java.sql.DatabaseMetaData 接口&#xff0c;我们能获取到数据库的列表、列等信息。 DatabaseMetaData 接口包含了许多方法&#xff0c;这里值介绍常用的。 ###获取 DatabaseMetaData 实例对象 DatabaseMetaData databaseMetaData connection.getMetaData(); 复制代码###获…

C++多线程(一)

C多线程&#xff08;一&#xff09; WIN 多线程API一 简单实例比较简单的代码&#xff0c;创建10个线程&#xff0c;其中使第4个线程在一创建就挂起&#xff0c;等到其他的线程执行的差不多的时候再使第4个线程恢复执行。#include <stdio.h>#include <stdlib.h>#i…

HALCON示例程序measure_ball_bond.hdev电路板焊点位置测量

HALCON示例程序measure_ball_bond.hdev电路板焊点位置测量 示例程序源码&#xff08;加注释&#xff09; 关于显示类函数解释 dev_update_off () dev_close_window () FileName : ‘bonds/ball_bond_ccd_’ read_image (Image, FileName 1$‘02’) dev_open_window_fit_imag…

rank()over 函数的使用

1. over()是分析函数&#xff0c;可以和rank()函数配合使用&#xff0c;也可以和其他函数配合使用。取每个学科排名前三的分数&#xff0c;sql语句如下&#xff1a; select * from (select rank() over(partition by subject order by mark desc) rk,S.* from S) T where T.rk&…

天梯赛2016-L2

L2-001. 紧急救援 作为一个城市的应急救援队伍的负责人&#xff0c;你有一张特殊的全国地图。在地图上显示有多个分散的城市和一些连接城市的快速道路。每个城市的救援队数量和每一条连接两个城市的快速道路长度都标在地图上。当其他城市有紧急求助电话给你的时候&#xff0c;你…

伺服系统控制网络的重要性! 现场总线的重要性! SSCNET运动控制系统与发展趋势

引言&#xff1a;在2010年的时候&#xff0c;在北京的一个数控公司工作。产品采用的是通过运动控制卡发脉冲的方式&#xff0c;控制机床的X、Y、Z轴进行加工。 机床在加工产品的时候&#xff0c;一直存在着精度的问题&#xff0c;例如DMG的机床可以达到0.01的加工精度&#x…

apache配置

wamp环境安装需要VC运行库的支持 apache如果需要外网访问&#xff0c;在其配置文件中寻找127.0.0.1然后替换为all 外网访问需要关闭防火墙&#xff1f;转载于:https://www.cnblogs.com/gremlin/p/5581486.html