主席树--动态区间第k小
主席树--动态区间第\(k\)小
模板题在这里洛谷2617。
先对几个问题做一个总结:
阅读本文需要有主席树的基础,也就是通过区间kth的模板题。
静态整体kth:
sort一下找第k小,时间复杂度\(O(nlogn)\)。
动态整体kth:
权值线段树维护一下,时间复杂度\(O(nlogn)\)。
静态区间kth:
主席树维护,时间复杂度\(O(nlogn)\)。
动态区间kth:
就是本次的标题。
回忆一下主席树是如何维护静态区间kth的。
建立可持久化线段树后,利用前缀和的思想查询区间的kth。
所以我们想对区间kth带修改操作,前缀和是关键。
我们在维护普通前缀和,支持查询和修改操作时,用的是什么数据结构呢?
- 树状数组/线段树。
所以这时候我们大致有一个概念了,动态主席树和主席树在数据结构上已经多少有点不一样了。
- 动态主席树:树套树。
- 静态主席树:可持久化权值线段树。
怎么套是一个问题,但简单想想可以发现,我们在外层维护一颗树状数组,树状数组的每个节点(内层)维护权值线段树(的根节点),可以解决这个问题。
- 修操作:
- 如果将一个位置的数字\(a_i=x\)修改为\(y\),那么在外层树状数组上,我们需要修改\(logn\)个节点,同时对于每个节点(代表了一颗权值线段树),分别有\(logn\)个节点受影响,所以修改复杂度为\(O((logn)^2)\)。
- 查操作:
- 对于每次\([L,R]\)之间的查询,我们先提取这区间的\(logn\)个根节点,然后将问题转换为静态主席树求区间第\(k\)小。
- 时间复杂度为\(O((logn)^2)\)。
所以总的算下来时间复杂度在\(O(n(logn)^2)\)上。
接下来解决一下空间的问题。
我们知道线段树的空间复杂度是\(O(4n)\)的,也就是\(O(n)\),树状数组的复杂度为\(O(n)\),那么如此算下来空间复杂度达到了\(O(n^2)\)。
思考一下怎么优化?
我们刚刚计算的空间复杂度,基于对每个节点都把完整的权值线段树开出来。
我们查询/修改操作的规模是\((logn)^2\)级别的。
那是不是可以动态开点。
对于我们能访问到的节点创立节点,对于不能访问到的,就不管了。
附\(luogu2617\)代码。
#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e5+10;
int n, m, a[maxn], b[maxn<<1], len;
struct Query
{
int op;
int i, j, k;
}q[maxn];
//1e5的log大概在20左右 20*20=400
int sum[maxn*400];
int ls[maxn*400];
int rs[maxn*400];
int rt[maxn*400];
int tot;
void update_SgT(int &rt, int l, int r, int x, int val)
{
if(!rt) rt = ++tot;
if(l == r)
{
sum[rt] += val;
return;
}
int mid = (l+r) >> 1;
if(x <= mid) update_SgT(ls[rt], l, mid, x, val);
else update_SgT(rs[rt], mid+1, r, x, val);
sum[rt] = sum[ls[rt]] + sum[rs[rt]];
}
inline int lowbit(int x){
return x&(-x);
}
void update_BIT(int pos, int x, int val)
{
for(int i = pos; i <= n; i += lowbit(i))
update_SgT(rt[i], 1, len, x, val);
}
///提取区间线段树的根节点
int rt1[maxn], rt2[maxn], cnt1, cnt2;
void locate(int l, int r)
{
cnt1 = cnt2 = 0;
for(int i = l-1; i; i -= lowbit(i))
rt1[++cnt1] = rt[i];
for(int i = r; i; i -= lowbit(i))
rt2[++cnt2] = rt[i];
}
int ask(int l, int r, int k)
{
if(l == r) return l;
int mid = (l+r) >> 1;
int suml = 0;
for(int i = 1; i <= cnt1; i++)
suml -= sum[ls[rt1[i]]];
for(int i = 1; i <= cnt2; i++)
suml += sum[ls[rt2[i]]];
if(suml >= k)
{
for(int i = 1; i <= cnt1; i++)
rt1[i] = ls[rt1[i]];
for(int i = 1; i <= cnt2; i++)
rt2[i] = ls[rt2[i]];
return ask(l, mid, k);
}
else
{
for(int i = 1; i <= cnt1; i++)
rt1[i] = rs[rt1[i]];
for(int i = 1; i <= cnt2; i++)
rt2[i] = rs[rt2[i]];
return ask(mid+1, r, k-suml);
}
}
int main()
{
scanf("%d%d", &n, &m);
for(int i = 1; i <= n; i++)
{
scanf("%d", &a[i]);
b[++len] = a[i];
}
char op[2];
for(int i = 1; i <= m; i++)
{
scanf("%s", op);
if(op[0] == 'Q')
{
q[i].op = 0;
scanf("%d%d%d", &q[i].i, &q[i].j, &q[i].k);
}
else
{
q[i].op = 1;
scanf("%d%d", &q[i].i, &q[i].k);
b[++len] = q[i].k;
}
}
//数值离散化
sort(b+1, b+1+len);
len = unique(b+1, b+1+len)-b-1;
for(int i = 1; i <= n; i++)
a[i] = lower_bound(b+1, b+len+1, a[i])-b;
for(int i = 1; i <= m; i++)
if(q[i].op) q[i].k = lower_bound(b+1, b+len+1, q[i].k)-b;
//建树(动态开点形式)
for(int i = 1; i <= n; i++)
update_BIT(i, a[i], 1);
for(int i = 1; i <= m; i++)
{
if(q[i].op)
{
update_BIT(q[i].i, a[q[i].i], -1);
a[q[i].i] = q[i].k;
update_BIT(q[i].i, q[i].k, 1);
}
else
{
locate(q[i].i, q[i].j);
int ans = b[ask(1, len, q[i].k)];
printf("%d\n", ans);
}
}
return 0;
}
主席树--动态区间第k小的更多相关文章
- A - 低阶入门膜法 - K-th Number (主席树查询区间第k小)
题目链接:https://cn.vjudge.net/contest/284294#problem/A 题目大意:主席树查询区间第k小. 具体思路:主席树入门. AC代码: #include<i ...
- HDU 5919 - Sequence II (2016CCPC长春) 主席树 (区间第K小+区间不同值个数)
HDU 5919 题意: 动态处理一个序列的区间问题,对于一个给定序列,每次输入区间的左端点和右端点,输出这个区间中:每个数字第一次出现的位子留下, 输出这些位子中最中间的那个,就是(len+1)/2 ...
- Super Mario HDU - 4417 (主席树询问区间比k小的个数)
Mario is world-famous plumber. His “burly” figure and amazing jumping ability reminded in our memory ...
- 主席树(区间第k小的数)
题目链接: https://www.luogu.org/problem/P3834 首先要离散化,然后主席树模板. 1 #include<cstdio> 2 #include<cst ...
- 洛谷.3834.[模板]可持久化线段树(主席树 静态区间第k小)
题目链接 //离散化后范围1~cnt不要错 #include<cstdio> #include<cctype> #include<algorithm> //#def ...
- 线段树维护区间前k小
线段树维护区间前k小 $ solution: $ 觉得超级钢琴太麻烦?在这里线段树提供一条龙服务 . 咳咳,开始讲正题!这道题我们有一个和超级钢琴复杂度一样 $ ~O(~\sum x\times lo ...
- poj 2104 主席树(区间第k大)
K-th Number Time Limit: 20000MS Memory Limit: 65536K Total Submissions: 44940 Accepted: 14946 Ca ...
- 主席树入门(区间第k大)
主席树入门 时隔5个月,我又来填主席树的坑了,现在才发现学算法真的要懂了之后,再自己调试,慢慢写出来,如果不懂,就只会按照代码敲,是不会有任何提升的,都不如不照着敲. 所以搞算法一定要弄清原理,和代码 ...
- 学习笔记--函数式线段树(主席树)(动态维护第K极值(树状数组套主席树))
函数式线段树..资瓷 区间第K极值查询 似乎不过似乎划分树的效率更优于它,但是如果主席树套树状数组后,可以处理动态的第K极值.即资瓷插入删除,划分树则不同- 那么原理也比较易懂: 建造一棵线段树(权值 ...
随机推荐
- class(二)--派生类的继承
前言 从我之前的一篇笔记对象的继承中, 我们可以知道JS的继承方式依赖原型链,而比较好的继承方式是寄生组合式继承 先来温习下什么是寄生组合式继承 function Rectangle(length, ...
- UVA - 1346 Songs (贪心+排序)
题意:已知每首歌的标号,长度和播放频率,求一种播放顺序,使得最小,并且输出该播放顺序下第t首歌的标号. 分析: 1.长度越短,播放频率越大的歌排在前面,上式越小. 2.s(i)表示的是当前播放顺序下这 ...
- 关于SSM中mybatis向oracle添加语句采用序列自增的问题
在SSM向oracle数据库中插入语句时,报错如下: ### Error updating database. Cause: java.sql.SQLException: 不支持的特性 ### SQ ...
- Vue - 项目配置 ( element , 安装路由 , 创建路由 )
1,安装element : vue add element 2,安装路由 : vue add router 3,创建路由的过程 : (1) 新建 vu ...
- dango 常用 静态文件 中间件 admin管理 上传图片
静态文件 项目中的CSS.图片.js都是静态文件.一般会将静态文件放到一个单独的目录中,以方便管理.在html页面中调用时,也需要指定静态文件的路径,Django中提供了一种解析的方式配置静态文件路径 ...
- 指令——touch
一个完整的指令的标准格式: Linux通用的格式——#指令主体(空格) [选项](空格) [操作对象] 一个指令可以包含多个选项,操作对象也可以是多个. 指令:touch 作用:创建文件 语法: ...
- 时间戳和LocalDateTime和Date互转和格式化
一 前言 续上篇java8在日常开发中使用LocalDate和LocalTime[https://blog.csdn.net/youku1327/article/details/102771936]中 ...
- centos7安装配置supervisor守护进程
yum install Supervisor supervisord -c /etc/supervisord.conf 进入 cd /etc 目录 找到supervisord.conf 配置文件 和 ...
- NumPy 排序、查找、计数
章节 Numpy 介绍 Numpy 安装 NumPy ndarray NumPy 数据类型 NumPy 数组创建 NumPy 基于已有数据创建数组 NumPy 基于数值区间创建数组 NumPy 数组切 ...
- ls 查看文件
1.按文件大小查看文件 a.降序:ls -lsh moudaen@morton:~$ ls -lshtotal 20M 20M -rw-r--r-- 1 moudaen 65536 20M Nov ...