Dynamic Rankings(树状数组套权值线段树)
Dynamic Rankings(树状数组套权值线段树)
给定一个含有n个数的序列a[1],a[2],a[3]……a[n],程序必须回答这样的询问:对于给定的i,j,k,在a[i],a[i+1],a[i+2]……a[j]中第k小的数是多少(1≤k≤j-i+1),并且,你可以改变一些a[i]的值,改变后,程序还能针对改变后的a继续回答上面的问题。你需要编一个这样的程序,从输入文件中读入序列a,然后读入一系列的指令,包括询问指令和修改指令。
对于每一个询问指令,你必须输出正确的回答。有两个正整数n(1≤n≤10000),m(1≤m≤10000)。分别表示序列的长度和指令的个数。
这道题就是一个带修区间第k大的问题,可以用cdq分治做,也可以用树套树做。由于有修改,如果暴力在主席树上修改,时间复杂度就爆炸了,单个修改都是\(O(nlogn)\)的。再看看查询,是\(O(logn)\)的,这就启发我们可不可以把它们两个平均一下,于是就用到了树状数组。在树状数组的每个区间上都建一颗权值线段树,维护此区间上每个值出现的个数(链+权值线段树=主席树,树状数组+权值线段树=带修主席树【滑稽】)。那么修改某一个数的值,会牵动到树状数组上logn个区间,然后权值线段树单点修改时间为\(O(logn)\),总时间是\(O(log^2nn)\)的。查询区间\([l, r]\)的第k大值,需要得到\([1, l-1]\)和\([1, r]\)的权值线段树,然后将它们相减,依然会牵动到logn个区间,由于权值线段树区间查询时间是\(O(logn)\),总时间也是\(O(logn^2n)\)的。注意要离散化,这题还是挺难写的。
所以说,数据结构可以平衡操作的时间复杂度。
P.S. 三个月后的我看到这篇博客仿佛看到了救星(要讲课啊妈蛋)
P.A. 四个月后的我忽然发现这个东西的本质和主席树的本质是不同的,因为主席树的每棵权值线段树是继承与前一棵权值线段树的,而它只是把插入和查询的前提复杂度变成了\(O(logn)\)而已。不过,我还是倾向于叫它带修主席树,因为它是主席树应用到树状数组上的自然推论。
#include <cctype>
#include <cstdio>
#include <algorithm>
using namespace std;
const int maxn=2e4+5;
int n, m, a[maxn], b[maxn], cntb, rt[maxn];
int c1[maxn], c2[maxn], c3[maxn];
int size[maxn*400], cseg, ls[maxn*400], rs[maxn*400];
int tset1[maxn], tset2[maxn], cnt1, cnt2;
inline int lowbit(int x){ return x&(-x); } //树状数组上的权值线段树
void ins(int &now, int l, int r, int pos, int v){
if (!now) now=++cseg; size[now]+=v;
if (l==r) return; int mid=(l+r)>>1;
if (pos<=mid) ins(ls[now], l, mid, pos, v);
else ins(rs[now], mid+1, r, pos, v);
}
void add(int x, int v){
int k=lower_bound(b+1, b+cntb+1, a[x])-b;
for (int i=x; i<=n; i+=lowbit(i))
ins(rt[i], 1, cntb, k, v);
}
int query(int l, int r, int v){ //logn个权值线段树一起跳
if (l==r) return l;
int sum=0, mid=(l+r)>>1;
for (int i=1; i<=cnt1; ++i) sum-=size[ls[tset1[i]]];
for (int i=1; i<=cnt2; ++i) sum+=size[ls[tset2[i]]];
if (v<=sum){
for (int i=1; i<=cnt1; ++i) tset1[i]=ls[tset1[i]];
for (int i=1; i<=cnt2; ++i) tset2[i]=ls[tset2[i]];
return query(l, mid, v);
} else {
for (int i=1; i<=cnt1; ++i) tset1[i]=rs[tset1[i]];
for (int i=1; i<=cnt2; ++i) tset2[i]=rs[tset2[i]];
return query(mid+1, r, v-sum);
}
}
inline void get(int &x){
char c; int flag=1;
for (; !isdigit(c=getchar()); ) if (c=='-') flag=-1;
for (x=c-48; c=getchar(), isdigit(c); )
x=(x<<3)+(x<<1)+c-48;
if (flag==-1) x=-x;
}
int main(){
get(n); get(m); char c;
for (int i=1; i<=n; ++i) get(a[i]), b[++cntb]=a[i];
for (int i=0; i<m; ++i){
while (!isgraph(c=getchar()));
get(c1[i]); get(c2[i]);
if (c=='Q') get(c3[i]); else b[++cntb]=c2[i];
}
sort(b+1, b+cntb+1); cntb=unique(b+1, b+cntb+1)-b-1;
for (int i=1; i<=n; ++i) add(i, 1);
for (int i=0; i<m; ++i) if (c3[i]){
cnt1=cnt2=0; //把要查询的区间都标出来
for (int j=c1[i]-1; j; j-=lowbit(j)) tset1[++cnt1]=rt[j];
for (int j=c2[i]; j; j-=lowbit(j)) tset2[++cnt2]=rt[j];
printf("%d\n", b[query(1, cntb, c3[i])]);
} else { add(c1[i], -1); a[c1[i]]=c2[i]; add(c1[i], 1); }
return 0;
}
Dynamic Rankings(树状数组套权值线段树)的更多相关文章
- [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)
[BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...
- BZOJ2141排队——树状数组套权值线段树(带修改的主席树)
题目描述 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家 乐和和.红星幼儿园的小朋友们排起了长长地队伍,准备吃果果.不过因为小朋友们的身高有所区别 ...
- luogu3380/bzoj3196 二逼平衡树 (树状数组套权值线段树)
带修改区间K大值 这题有很多做法,我的做法是树状数组套权值线段树,修改查询的时候都是按着树状数组的规则找出那log(n)个线段树根,然后一起往下做 时空都是$O(nlog^2n)$的(如果离散化了的话 ...
- CF1093E Intersection of Permutations 树状数组套权值线段树
\(\color{#0066ff}{ 题目描述 }\) 给定整数 \(n\) 和两个 \(1,\dots,n\) 的排列 \(a,b\). \(m\) 个操作,操作有两种: \(1\ l_a\ r_a ...
- 【树状数组套权值线段树】bzoj1901 Zju2112 Dynamic Rankings
谁再管这玩意叫树状数组套主席树我跟谁急 明明就是树状数组的每个结点维护一棵动态开结点的权值线段树而已 好吧,其实只有一个指针,指向该结点的权值线段树的当前结点 每次查询之前,要让指针指向根结点 不同结 ...
- 刷题总结——骑士的旅行(bzoj4336 树链剖分套权值线段树)
题目: Description 在一片古老的土地上,有一个繁荣的文明. 这片大地几乎被森林覆盖,有N座城坐落其中.巧合的是,这N座城由恰好N-1条双 向道路连接起来,使得任意两座城都是连通的.也就是说 ...
- 【bzoj3065】带插入区间K小值 替罪羊树套权值线段树
题目描述 从前有n只跳蚤排成一行做早操,每只跳蚤都有自己的一个弹跳力a[i].跳蚤国王看着这些跳蚤国欣欣向荣的情景,感到非常高兴.这时跳蚤国王决定理性愉悦一下,查询区间k小值.他每次向它的随从伏特提出 ...
- 「ZJOI2017」树状数组(二维线段树)
「ZJOI2017」树状数组(二维线段树) 吉老师的题目真是难想... 代码中求的是 \(\sum_{i=l-1}^{r-1}a_i\),而实际求的是 \(\sum_{i=l}^{r}a_i\),所以 ...
- HDU 1934 树状数组 也可以用线段树
http://acm.hdu.edu.cn/showproblem.php?pid=1394 或者是我自己挂的专题http://acm.hust.edu.cn/vjudge/contest/view. ...
随机推荐
- DELPHI中四种EXCEL访问技术实现
一.引言 EXCEL在处理中文报表时功能非常强大,EXCEL报表访问也是信息系统开发中的一个重要内容,本文总结以往开发中所用到的几中EXCEL文件访问方法,在实际工作中也得到了很好的验证,本文列举了其 ...
- AngularJS学习笔记(二) 表单验证案例(ng-repeat/filter)
这一节相对来说需要理解的东西不是太多,记住了那些api就行了. 还是一个案例(同样来自miaov),一个表单验证,先上代码,然后再对对应的内容进行解释. <!DOCTYPE html> & ...
- Python入门经典练习题
[程序1] 题目:有1.2.3.4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少? num_list=[]cou=0for i in range(1,5): for j in range( ...
- Jmete基础使用
1,jmeter下载与安装 Jmeter的运行需要JDK支持,所以需要先安装好jdk,并配置好环境变量: 下载地址:http://jmeter.apache.org/download_jmeter.c ...
- 51nod 1829(函数)
题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1829 本题目相当于: n个不同的小球,放入到m个可区分的盒子 ...
- 关于MFC主菜单和右键弹出菜单
一.主菜单.弹出菜单和右键菜单的概念: 主菜单是窗口顶部的菜单,一个窗口或对话框只能有一个主菜单,但是主菜单可以被更改(SetMenu()更改): 创建方式:CMenu::CreateMenu(voi ...
- linux命令学习笔记(17):whereis 命令
whereis命令只能用于程序名的搜索,而且只搜索二进制文件(参数-b).man说明文件(参数-m)和 源代码文件(参数-s).如果省略参数,则返回所有信息. 和find相比,whereis查找的速度 ...
- ACM学习历程—HDU1695 GCD(容斥原理 || 莫比乌斯)
Description Given 5 integers: a, b, c, d, k, you're to find x in a...b, y in c...d that GCD(x, y) = ...
- UnityShader实例15:屏幕特效之Bloom
http://blog.csdn.net/u011047171/article/details/48522073 Bloom特效 概述 Bloom,又称“全屏泛光”,是游戏中 ...
- NSDictionary 用法
//Dictionary //不可变 //字典里面:是以键值对存放的 //字典存放顺序是无序的 //字典里面:value可以重复,但key不能重复 //字典里面:key一般用字符串表示,value可用 ...