【序列操作IV】树状数组套线段树/树套树
题目描述
给出序列 a1,a2,…,an(0≤ai≤109),有关序列的两种操作。
1. ai(1≤i≤n)变成 x(0≤x≤109)。
2. 求 al,al+1,…,ar(1≤l≤r≤n)第 k(1≤k≤r-l+1)小。
输入格式
第一行包含两个数 n(1≤n≤2×104)和 m(1≤m≤2×104),表示序列长度和操作次数。
接下来一行 n 个数,以空格隔开,表示 a1,a2,…,an 。
接下来m行,每行为以下两种格式之一:
- 0 i x , 表示 ai=x。
- 1 l r k ,求 al,al+1,…,ar 的第 k 小。
输出格式
对于每次询问,输出单独的一行表示答案。
样例数据 1
输入
5 3
1 2 3 4 5
1 1 5 3
0 3 5
1 1 5 3
输出
3
4
题目分析
本题有多种解法。这里先讲一下较为简单的树状数组套线段树。
外层树状数组n个节点分别对应下标,每个节点其实是一个权值为下标的线段树的根节点(一共有n颗),那么树状数组中的i号点表示的就是区间为1~i的元素,而其下的权值线段树则记录在该区间的权值出现次数。
由于树状数组有前缀和的性质,插入时会更新它及其以后的所有树,询问时也能访问它及其以前的所有树,所以要查询L~R的k小,就只需要将有关区间的点放入一个数组,然后分别进入左子树、右子树查询即可。
看看代码很好理解。
code
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<cmath>
#include<cstdlib>
#include<algorithm>
using namespace std; const int N = 2e4 + , M = 2e6 + ;
int len, a[N], b[N * ], root[N], tot;
int cur[N], n, m;
struct node{
int cnt, lc, rc;
}tr[M];
struct qry{
int opt, a, b, c;
}Q[N]; inline int read(){
int i = , f = ; char ch = getchar();
for(; (ch < '' || ch > '') && ch != '-'; ch = getchar());
if(ch == '-') f = -, ch = getchar();
for(; ch >= '' && ch <= ''; ch = getchar())
i = (i << ) + (i << ) + (ch - '');
return i * f;
} inline void wr(int x){
if(x < ) putchar('-'), x = -x;
if(x > ) wr(x / );
putchar(x % + '');
} inline void disc_init(){
sort(b + , b + len + );
len = unique(b + , b + len + ) - (b + );
for(int i = ; i <= n; i++)
a[i] = lower_bound(b + , b + len + , a[i]) - b;
for(int i = ; i <= m; i++)
if(Q[i].opt == ) Q[i].b = lower_bound(b + , b + len + , Q[i].b) - b;
}
inline void insert(int &k, int l, int r, int p, int v){
if(!k) k = ++tot;
tr[k].cnt += v;
if(l == r) return;
int mid = l + r >> ;
if(p <= mid) insert(tr[k].lc, l, mid, p, v);
else insert(tr[k].rc, mid + , r, p, v);
} inline void update(int k, int p, int v){
for(int i = k; i <= n; i += (i & -i))
insert(root[i], , len, p, v);
} inline void gotoPos(int lx, int rx){
for(int i = rx; i > ; i -= (i & -i)) cur[i] = root[i];
for(int i = lx; i > ; i -= (i & -i)) cur[i] = root[i];
} inline int CntLc(int lx, int rx){
int ret = ;
for(int i = rx; i > ; i -= (i & -i)) ret += tr[tr[cur[i]].lc].cnt;
for(int i = lx; i > ; i -= (i & -i)) ret -= tr[tr[cur[i]].lc].cnt;
return ret;
} inline int query(int nl, int nr, int l, int r, int k){
if(l == r) return l;
int ret = CntLc(nl, nr), mid = l + r >> ;
if(ret >= k){
for(int i = nr; i > ; i -= (i & -i)) cur[i] = tr[cur[i]].lc;
for(int i = nl; i > ; i -= (i & -i)) cur[i] = tr[cur[i]].lc;
return query(nl, nr, l, mid, k);
}
else{
for(int i = nr; i > ; i -= (i & -i)) cur[i] = tr[cur[i]].rc;
for(int i = nl; i > ; i -= (i & -i)) cur[i] = tr[cur[i]].rc;
return query(nl, nr, mid + , r, k - ret);
}
} inline void tree_init(){
for(int i = ; i <= n; i++) update(i, a[i], );
} int main(){
n = read(), m = read();
for(int i = ; i <= n; i++) a[i] = b[++len] = read();
for(int i = ; i <= m; i++){
Q[i].opt = read();
if(Q[i].opt == )
Q[i].a = read(), Q[i].b = read(), Q[i].c = read();
else Q[i].a = read(), Q[i].b = b[++len] = read();
}
disc_init();
tree_init();
for(int i = ; i <= m; i++){
if(Q[i].opt == ){
update(Q[i].a, a[Q[i].a], -);
a[Q[i].a] = Q[i].b;
update(Q[i].a, Q[i].b, );
}
else{
gotoPos(Q[i].a - , Q[i].b);
wr(b[query(Q[i].a - , Q[i].b, , len, Q[i].c)]), putchar('\n');
}
}
}
【序列操作IV】树状数组套线段树/树套树的更多相关文章
- 【BZOJ3196】二逼平衡树(树状数组,线段树)
[BZOJ3196]二逼平衡树(树状数组,线段树) 题面 BZOJ题面 题解 如果不存在区间修改操作: 搞一个权值线段树 区间第K大--->直接在线段树上二分 某个数第几大--->查询一下 ...
- HDU 5877 2016大连网络赛 Weak Pair(树状数组,线段树,动态开点,启发式合并,可持久化线段树)
Weak Pair Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others) Tota ...
- st表、树状数组与线段树 笔记与思路整理
已更新(2/3):st表.树状数组 st表.树状数组与线段树是三种比较高级的数据结构,大多数操作时间复杂度为O(log n),用来处理一些RMQ问题或类似的数列区间处理问题. 一.ST表(Sparse ...
- POJ 1195 Mobile phones (二维树状数组或线段树)
偶然发现这题还没A掉............速速解决了............. 树状数组和线段树比较下,线段树是在是太冗余了,以后能用树状数组还是尽量用......... #include < ...
- HDU 5618 Jam's problem again(三维偏序,CDQ分治,树状数组,线段树)
Jam's problem again Time Limit: 5000/2500 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Othe ...
- Codeforces 777E(离散化+dp+树状数组或线段树维护最大值)
E. Hanoi Factory time limit per test 1 second memory limit per test 256 megabytes input standard inp ...
- HYSBZ - 3813 奇数国 欧拉函数+树状数组(线段树)
HYSBZ - 3813奇数国 中文题,巨苟题,巨无敌苟!!首先是关于不相冲数,也就是互质数的处理,欧拉函数是可以求出互质数,但是这里的product非常大,最小都2100000,这是不可能实现的.所 ...
- 洛谷P3374 【模板】树状数组 1&&P3368 【模板】树状数组 2题解
图片来自度娘~~ 树状数组形如上图,是一种快速查找区间和,快速修改的一种数据结构,一个查询和修改复杂度都为log(n),树状数组1和树状数组2都是板子题,在这里进行详解: 求和: 首先我们看一看这个图 ...
- 洛谷P3380 【模板】二逼平衡树(树套树,树状数组,线段树)
洛谷题目传送门 emm...题目名写了个平衡树,但是这道题的理论复杂度最优解应该还是树状数组套值域线段树吧. 就像dynamic ranking那样(蒟蒻的Sol,放一个link骗访问量233) 所有 ...
- ZJOI 2017 树状数组(线段树套线段树)
题意 http://uoj.ac/problem/291 思路 不难发现,九条カレン醬所写的树状数组,在查询区间 \([1,r]\) 的时候,其实在查询后缀 \([r,n]\) :在查询 \([l,r ...
随机推荐
- 【2017 Multi-University Training Contest - Team 10】Schedule
[链接]http://acm.hdu.edu.cn/contests/contest_showproblem.php?pid=1010&cid=767 [题意] 给一些区间,每台机器在这些区间 ...
- socket长连接的维持
长连接的维持,是要客户端程序,定时向服务端程序,发送一个维持连接包的.如果,长时间未发送维持连接包,服务端程序将断开连接. 客户端:通过持有Client对象,可以随时(使用sendObject方法)发 ...
- Android 解决RecyclerView删除Item导致位置错乱的问题
RecyclerView的刷新分为内容变化和结构变化,结构变化比如remove和insert等并不会导致viewholder的更新,所以有时候我们使用 notifyItemRemoved(positi ...
- 【2017 ACM/ICPC 乌鲁木齐赛区网络赛环境测试赛 E】蒜头君的排序
[链接]h在这里写链接 [题意] 在这里写题意 [题解] 莫队算法+树状数组. 区间增加1或减少1. 对逆序对的影响是固定的. (用冒泡排序变成升序的交换次数,就是逆序对的个数) [错的次数] 0 [ ...
- mahout測试朴素贝叶斯分类样例
对于这个測试建议大家先理解原理,这里我画了例如以下的示意图 接下来就依照例如以下的细节来输入指令測试: 首先前提是Hadoop安装并启动,mahout已经安装了. <strong>< ...
- iOS_02_什么是ios开发
什么是ios开发? * 已知:ios是iphone,ipad等手持设备操作系统. * ios开发就是开发运行在ios系统上的应用或者游戏软件,比如手机QQ,微博或者游戏,说白了,就是开发手机软件:当然 ...
- 每日技术总结:promise,express route,评分,local storage商品浏览历史,
最近正在用Vue做一个电商项目.利用工作前后空隙时间. 1.promise的使用 点这里 如何在实际项目中使用Promise 2. Express Route 前后端传参的两种方法 (1)req.pa ...
- Microsoft iSCSI Software Target 快照管理
Microsoft iSCSI Software Target 支持快照管理,可以对设备进行手工创建快照.快照任务计划.快照回滚等操作. 首先配置iscsi 目标及设备映射关系,并在客户端通过发起程序 ...
- Aamazon Web Service EC2 Ubuntu 新建用户而且用ssh连接host
本文參照 http://docs.aws.amazon.com/zh_cn/AWSEC2/latest/UserGuide/managing-users.html http://docs.aws.am ...
- 一文看懂AI芯片竞争五大维度
下一波大趋势和大红利从互联网+让位于人工智能+,已成业界共识.在AI的数据.算法和芯片之三剑客中,考虑到AI算法开源的发展趋势,数据与芯片将占据越来越重要的地位,而作为AI发展支柱的芯片更是AI业的竞 ...