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. ...
随机推荐
- java支付宝开发-异常-01-"sub_code":"isv.invalid-app-id","sub_msg":"无效的AppID参数"
一.现象 无论请求哪个接口都报这个错误 二.异常原因 后来检查了一下,发现是因为 我支付宝网关写错了.沙箱环境和正式环境 的支付宝网关不同,如下 //支付宝网关名-正式环境 //public stat ...
- web项目路径如何更改
- C++ 值传递、址传递、引用传递
一.值传递 int func(int p) 值传递会在栈中开辟一块空间 p,使得p和实参的a 同值. 此时你在函数func里面对p进行任何操作都不会对原值a产生任何影响.因为a 和p本就就是两个变 ...
- CI中控制器名不能和本个 控制器中的方法名相同
控制器名称:application/controllers/tang.php 控制器中方法名称:application/controllers/role.php 中有方法 public funct ...
- Agc011_C Squared Graph
传送门 题目大意 给定$n$个点$m$条边的简单图(无重边无自环),将有序点对$\{a,b\}$作为新的点,新产生的$n^2$个点中对于两个点,$\{a,b\},\{x,y\}$,当且仅当原图中存在边 ...
- Mybatis generator配置文件及说明
项目采用sring mvc + mybatis 组合,这里简单介绍下mybatis的应用: 我的IDE是STS(Spring + Tool + Suite), 安装Mybatis Generator插 ...
- 错误名称:Uncaught SyntaxError: Unexpected identifier
控制台输出: 1.谷歌:Uncaught SyntaxError: Unexpected identifier 2.火狐:SyntaxError: missing ] after element li ...
- BestCoder Round #92 比赛记录
上午考完试后看到了晚上的BestCoder比赛,全机房都来参加 感觉压力好大啊QAQ,要被虐了. 7:00 比赛开始了,迅速点进了T1 大呼这好水啊!告诉了同桌怎么看中文题面 然后就开始码码码,4分1 ...
- unity中mesh属性的uv坐标讨论
http://blog.sina.com.cn/s/blog_427cf00b0102vp0j.html 之前在做连连看游戏中,也用到贴图坐标,当时我们讲到,不管是平铺(Tiling)还是偏移(Off ...
- 快速排序的JavaScript实现
思想 分治的思想,将原始数组分为较小的数组(但没有像归并排序一样将它们分隔开). 主元选择: 从数组中任意选择一项作为主元,通常为数组的第一项,即arr[i]:或数组的中间项, arr[Math.fl ...