区间第k大的几种解法
区间第K大问题,变化包括带修改和不带修改,强制在线和允许离线
修改主要是单点修改,我们前面也只讨论这种情况。
接下来我们从编程复杂度和时空复杂度来讨论几种做法。
1.整体二分(编程复杂度:低-中,时间复杂度:优秀,空间复杂度:优秀)
缺点:只能做离线
优点:空间都是O(n)。常数小。带修改O(nlog2n),不带修改O(nlogn)。
但是不带修改的情况,如果允许的话,个人认为加个树状数组写O(nlog2n)的更好写
这时单次solve中面对的问题是,数列中一些点是1其余都是0,然后求区间和的问题
这个问题把区间和变成前缀和相减,然后用vector来对那些需要求前缀的点做桶排
然后双指针即可做到单次solve为O(k),k为此次处理的操作数,总体O(nlogn)的复杂度
显然如果不差那log的时间,直接用树状数组来处理更好写
下面给了一个带单点修改查询区间第K小的整体二分代码
#include <bits/stdc++.h> #define lb(x) (x&(-x)) using namespace std; const int N = 2e5 + ; int t, n, m, k, cnt; struct node {
int id, i, j, k;
}a[N], q1[N], q2[N]; int ans[N], b[N]; int c[N]; char op[]; void add(int i, int x) {while (i <= n) c[i] += x, i += lb(i);} int ask(int i) {int res = ; while (i > ) res += c[i], i -= lb(i); return res;} void solve(int head, int tail, int l, int r) {
if (head > tail) return;
if (l == r) {
for (int i = head; i <= tail; i ++)
ans[a[i].id] = r;
return;
}
int mid = l + r >> , s1 = , s2 = ;
for (int sum, i = head; i <= tail; i ++)
if (a[i].id) {
sum = ask(a[i].j) - ask(a[i].i - );
if (sum >= a[i].k) q1[s1 ++] = a[i];
else a[i].k -= sum, q2[s2 ++] = a[i];
}
else {
if (a[i].j <= mid) q1[s1 ++] = a[i], add(a[i].i, a[i].k);
else q2[s2 ++] = a[i];
}
for (int i = ; i < s1; i ++)
if (!q1[i].id)
add(q1[i].i, -q1[i].k);
memcpy(a + head, q1, sizeof(node) * s1);
memcpy(a + head + s1, q2, sizeof(node) * s2);
solve(head, head + s1 - , l, mid);
solve(head + s1, tail, mid + , r);
} int main() {
ios::sync_with_stdio(false);
for (cin >> t; t --; ) {
cin >> n >> m; k = cnt = ;
for (int i = ; i <= n; i ++) {
cin >> b[i];
a[++ k] = (node){, i, b[i], };
}
for (int l, r, x, i = ; i <= m; i ++) {
cin >> op >> l >> r;
if (op[] == 'Q') {
cin >> x;
a[++ k] = (node){++ cnt, l, r, x};
}
else {
a[++ k] = (node){, l, b[l], -};
a[++ k] = (node){, l, b[l] = r, };
}
}
solve(, k, , 1e9);
for (int i = ; i <= cnt; i ++)
printf("%d\n", ans[i]);
}
return ;
}
2.主席树(编程复杂度:低-中,时间复杂度:优秀,空间复杂度:高)
缺点:空间占用多。树套树的常数。
优点:可以在线!时间复杂度同整体二分。空间复杂度和时间复杂度一致。
不带修改是主席树基本操作。带修改就套树状数组,下面给出(和上面同一个问题的)代码。
#include <bits/stdc++.h> using namespace std; const int MAXN = 1e9;
const int N = 5e4 + ; int n, m, a[N], rt[N]; int tot, tr[N * ][]; int tmp1[], tmp2[]; #define l(x) tr[x][0]
#define r(x) tr[x][1]
#define s(x) tr[x][2]
#define lb(x) (x&(-x))
#define mid (l + r >> 1) int change(int o, int l, int r, int k, int v) {
int x = ++ tot; s(x) = s(o) + v;
if (l == r) return x; l(x) = l(o), r(x) = r(o);
k > mid ? r(x) = change(r(o), mid + , r, k, v) : l(x) = change(l(o), l, mid, k, v);
return x;
} void modify(int i, int p, int v) {
while (i <= n) rt[i] = change(rt[i], , MAXN, p, v), i += lb(i);
} int ask(int l, int r, int k) {
if (l == r) return r; int sum = ;
for (int i = ; i <= tmp1[]; i ++) sum -= s(l(tmp1[i]));
for (int i = ; i <= tmp2[]; i ++) sum += s(l(tmp2[i]));
if (k > sum) {
for (int i = ; i <= tmp1[]; i ++) tmp1[i] = r(tmp1[i]);
for (int i = ; i <= tmp2[]; i ++) tmp2[i] = r(tmp2[i]);
return ask(mid + , r, k - sum);
}
else {
for (int i = ; i <= tmp1[]; i ++) tmp1[i] = l(tmp1[i]);
for (int i = ; i <= tmp2[]; i ++) tmp2[i] = l(tmp2[i]);
return ask(l, mid, k);
}
} int query(int l, int r, int k) {//查询区间第k小
tmp1[] = tmp2[] = ;
for (int i = l - ; i > ; i -= lb(i)) tmp1[++ tmp1[]] = rt[i];
for (int i = r; i > ; i -= lb(i)) tmp2[++ tmp2[]] = rt[i];
return ask(, MAXN, k);
} int main(){
int t; char op[];
for (cin >> t; t --; ) {
cin >> n >> m; tot = ;
for (int i = ; i <= n; i ++) cin >> a[i], rt[i] = ;
for (int i = ; i <= n; i ++) modify(i, a[i], );
for (int i, j, k; m --; ) {
cin >> op >> i >> j;
if (op[] == 'C') modify(i, a[i], -), modify(i, a[i] = j, );
else cin >> k, printf("%d\n", query(i, j, k));
}
}
return ;
}
其他做法我参考了一下,很难与上述两种做法并肩,就不做讨论了
如果有的话欢迎告诉我
拓展1.初始数列每个位置都是一个空队列,修改变成了区间[l,r]的每个队列末尾都push一个数x
查询某个区间所有数字中的第K大
解法:其实还是个整体二分的简单题目,用线段树维护即可,复杂度依然是O(nlog2n)
区间第k大的几种解法的更多相关文章
- 静态区间第k大 树套树解法
然而过不去你谷的模板 思路: 值域线段树\([l,r]\)代表一棵值域在\([l,r]\)范围内的点构成的一颗平衡树 平衡树的\(BST\)权值为点在序列中的位置 查询区间第\(k\)大值时 左区间在 ...
- 【POJ】【2104】区间第K大
可持久化线段树 可持久化线段树是一种神奇的数据结构,它跟我们原来常用的线段树不同,它每次更新是不更改原来数据的,而是新开节点,维护它的历史版本,实现“可持久化”.(当然视情况也会有需要修改的时候) 可 ...
- HDU3473--Minimum Sum(静态区间第k大)
Minimum Sum Time Limit: 16000/8000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Tota ...
- 解决区间第K大的问题的各种方法
例题:http://poj.org/problem?id=2104 最近可能是念念不忘,必有回响吧,总是看到区间第k大的问题,第一次看到是在知乎上有人面试被弄懵了后来又多次在比赛中看到.以前大概是知道 ...
- Dynamic Rankings——带修改区间第k大
三种做法:1.整体二分: 二分mid 考虑小于mid的修改的影响 但是大于mid的修改可能会干掉小于mid的一些值 所以额外把一个修改变成一个值的删除和一个值的添加 这样就相互独立了! 整体二分,树状 ...
- Permutation UVA - 11525(值域树状数组,树状数组区间第k大(离线),log方,log)(值域线段树第k大)
Permutation UVA - 11525 看康托展开 题目给出的式子(n=s[1]*(k-1)!+s[2]*(k-2)!+...+s[k]*0!)非常像逆康托展开(将n个数的所有排列按字典序排序 ...
- POJ 2104 静态找区间第k大
静态区间第k大的问题,往往可以利用主席树来解决 这是主席树的第一道题 主席树大概可以理解为在n个节点上都建立一棵线段树,但是想想会超出内存 每一个节点保存的线段树都记录当前整段前缀区间的信息 但是因为 ...
- 【ZOJ2112】【整体二分+树状数组】带修改区间第k大
The Company Dynamic Rankings has developed a new kind of computer that is no longer satisfied with t ...
- 【POJ2761】【区间第k大】Feed the dogs(吐槽)
Description Wind loves pretty dogs very much, and she has n pet dogs. So Jiajia has to feed the dogs ...
随机推荐
- 【C++11新特性】 C++11智能指针之shared_ptr
C++中的智能指针首先出现在“准”标准库boost中.随着使用的人越来越多,为了让开发人员更方便.更安全的使用动态内存,C++11也引入了智能指针来管理动态对象.在新标准中,主要提供了shared_p ...
- IDEA使用中的快捷键
项目与项目之间的跳转: Ctrl+Alt+] 下一个窗口. Ctrl+Alt+[ 跳转回上一个窗口. 文件之间的跳转: Ctrl+E. ...
- DOS基础使用专题(强烈推荐)
DOS基础使用专题(强烈推荐) 美丽的DOS时代 DOS是世界上使用人数最多的操作系统,包括上面的Win3.x/9x等GUI操作平台的用户.尽管许多人由于种种原因而使用了其它非DOS的操作系统或操作环 ...
- javascript中new关键字详解
和其他高级语言一样 javascript 中也有 new 运算符,我们知道 new 运算符是用来实例化一个类,从而在内存中分配一个实例对象. 但在 javascript 中,万物皆对象,为什么还要通过 ...
- 专家揭秘:STM32启动过程全解
电子发烧友网核心提示:本文主要阐述了STM32启动过程全面解析,包括启动过程的介绍.启动代码的陈列以及深入解析. 相对于ARM上一代的主流ARM7/ARM9内核架构,新一代Cortex内核架构的启动方 ...
- Linux ftp安装
ftp安装部分,操作步骤如下: 可以使用yum命令直接安装ftp # yum install vsftpd ftp服务的开启与关闭命令: 开启:# /bin/systemctl start vsftp ...
- Yahoo34条军规——雅虎WEB前端网站优化
雅虎给出了优化网站加载速度的34条法则(包括Yslow规则22条) 详细说明,下载转发 ponytail 的译文(来自帕兰映像). 1.Minimize HTTP Requests 减少HTTP请求 ...
- mybatis 学习视频总结记录
学习mybaits简单增删改查例子记录 此整理是学习视频后的视频内容整理,后半段还没有整理 网易云课堂 SSM高级整合视频 地址 : http://study.163.com/course/cours ...
- 4.jmeter在线并发的怎样设置
4.1Jmeter 快速入门教程(一) - 认识jmeter和google插件 4.2Jmeter 快速入门教程(二)--创建简单web测试 打印 E-mail 4.3Jmeter 快速入门教程(三- ...
- v8引擎的优化
1.编译优化 V8采用JIT即使编译技术. 例如JAVA是先编译成字节码,再由JVM编译成机器码,V8则没有中间的字节码,直接由源码生成语法树,然后编译成机器码. 2.隐藏类 当定义一个构造函数,使用 ...