题面

单点修改,区间求第k大

分析

首先,这道题卡权值线段树套treap的做法,所以只能用主席树做

对于静态的查询,root[i]对应的主席树的区间[l,r]保存的是a[1]a[i]有多少个值落在区间[l,r]内。如果我们要修改a[i],则要修改O(n)棵主席树。那么我们像树状数组那样维护n棵主席树,不同的是每棵主席树里保存的是,a[i-lowbit(i)+1]a[i]有多少个值落在区间[l,r]内

对于查询[ql,qr]时的做差

我们要像树状数组求和那样,把root[i],root[i-lowbit(i)],....共O(logn)棵主席树的值加起来,才能得到a[1]a[i]有多少个数落在1qr里面,1~(ql-1)同理,维护两个临时数组x,y存储这O(logn)棵主席树的根即可

对于单点修改,我们像树状数组修改那样,往O(logn)棵主席树中插入即可

时间复杂度\(O(n \log ^2 n)\),空间复杂度略大,为\(O(n\log^2n)\)

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<ctime>
#include<algorithm>
#define maxn 200005
#define maxlogn 27
using namespace std;
inline void qread(int& x) {
x=0;
char c=getchar();
int sign=1;
while(c<'0'||c>'9') {
if(c=='-') sign=-1;
c=getchar();
}
while(c>='0'&&c<='9') {
x=x*10+c-'0';
c=getchar();
}
x*=sign;
}
inline void qread(char &x){
char c=getchar();
while(c<'A'||c>'Z'){
c=getchar();
}
x=c;
}
inline void qprint(int x) {
if(x<0) {
putchar('-');
qprint(-x);
} else if(x==0) {
putchar('0');
return;
} else {
if(x/10>0) qprint(x/10);
putchar('0'+x%10);
}
} int n,m;
struct node {
int ls;
int rs;
int sum;
} tree[maxn*maxlogn*maxlogn];
int root[maxn*maxlogn];
int ptr;
void push_up(int x) {
tree[x].sum=tree[tree[x].ls].sum+tree[tree[x].rs].sum;
}
void update(int &x,int upos,int uval,int last,int l,int r) {
x=++ptr;
tree[x]=tree[last];
if(l==r) {
tree[x].sum+=uval;
return;
}
int mid=(l+r)>>1;
if(upos<=mid) update(tree[x].ls,upos,uval,tree[last].ls,l,mid);
else update(tree[x].rs,upos,uval,tree[last].rs,mid+1,r);
push_up(x);
}
int totx,x[maxn*maxlogn],toty,y[maxn*maxlogn];
int query(int k,int l,int r) {
if(l==r) return l;
int lcnt=0;
int mid=(l+r)>>1;
for(int i=1;i<=totx;i++){
lcnt-=tree[tree[x[i]].ls].sum;
}
for(int i=1;i<=toty;i++){
lcnt+=tree[tree[y[i]].ls].sum;
}
if(k<=lcnt){
for(int i=1;i<=totx;i++) x[i]=tree[x[i]].ls;
for(int i=1;i<=toty;i++) y[i]=tree[y[i]].ls;
return query(k,l,mid);
}else{
for(int i=1;i<=totx;i++) x[i]=tree[x[i]].rs;
for(int i=1;i<=toty;i++) y[i]=tree[y[i]].rs;
return query(k-lcnt,mid+1,r);
}
}
inline int lowbit(int x){
return x&-x;
} struct oper {
int type;
int l,r,k;
int pos,val;
int dval;
} q[maxn];
int a[maxn];
int sz;
int b[maxn*2]; void add(int xpos,int xval,int v){
for(int i=xpos;i<=sz;i+=lowbit(i)){
update(root[i],xval,v,root[i],1,sz);
}
}
int answer(int l,int r,int k){
totx=toty=0;
for(int i=l-1;i;i-=lowbit(i)) x[++totx]=root[i];
for(int i=r;i;i-=lowbit(i)) y[++toty]=root[i];
return query(k,1,sz);
}
int main() {
char cmd;
qread(n);
qread(m);
for(int i=1; i<=n; i++) {
qread(a[i]);
b[++sz]=a[i];
}
for(int i=1; i<=m; i++) {
qread(cmd);
if(cmd=='Q') {
q[i].type=0;
qread(q[i].l);
qread(q[i].r);
qread(q[i].k);
} else {
q[i].type=1;
qread(q[i].pos);
qread(q[i].val);
b[++sz]=q[i].val;
}
}
sort(b+1,b+1+sz);
sz=unique(b+1,b+1+sz)-b-1;
for(int i=1; i<=n; i++) a[i]=lower_bound(b+1,b+1+sz,a[i])-b;
for(int i=1; i<=m; i++) {
if(q[i].type==1) {
q[i].dval=lower_bound(b+1,b+1+sz,q[i].val)-b;
}
}
for(int i=1; i<=n; i++) {
add(i,a[i],1);
}
// T2.debug(1);
for(int i=1; i<=m; i++) {
if(q[i].type==0) {
qprint(b[answer(q[i].l,q[i].r,q[i].k)]);
putchar('\n');
} else {
add(q[i].pos,a[q[i].pos],-1);
add(q[i].pos,q[i].dval,1);
a[q[i].pos]=q[i].dval; // T2.debug(1);
}
}
}

[BZOJ1901][luogu2617]Dynamic Rankings(树状数组+主席树)的更多相关文章

  1. BZOJ_1901_Zju2112 Dynamic Rankings_树状数组+主席树

    BZOJ_1901_Zju2112 Dynamic Rankings_树状数组+主席树 题意: 给定一个含有n个数的序列a[1],a[2],a[3]……a[n],程序必须回答这样的询问:对于给定的i, ...

  2. zoj2112 树状数组+主席树 区间动第k大

    Dynamic Rankings Time Limit: 10000MS   Memory Limit: 32768KB   64bit IO Format: %lld & %llu Subm ...

  3. 【bzoj1146】[CTSC2008]网络管理Network 倍增LCA+dfs序+树状数组+主席树

    题目描述 M公司是一个非常庞大的跨国公司,在许多国家都设有它的下属分支机构或部门.为了让分布在世界各地的N个部门之间协同工作,公司搭建了一个连接整个公司的通信网络.该网络的结构由N个路由器和N-1条高 ...

  4. 【bzoj3744】Gty的妹子序列 分块+树状数组+主席树

    题目描述 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzoj3720) 上掉落下来了许多妹子,他发现 她们排成 ...

  5. BZOJ_2120_数颜色_Set+树状数组+主席树

    BZOJ_2120_数颜色_Set+树状数组+主席树 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L ...

  6. P1972 [SDOI2009]HH的项链[离线+树状数组/主席树/分块/模拟]

    题目背景 无 题目描述 HH 有一串由各种漂亮的贝壳组成的项链.HH 相信不同的贝壳会带来好运,所以每次散步完后,他都会随意取出一段贝壳,思考它们所表达的含义.HH 不断地收集新的贝壳,因此,他的项链 ...

  7. [luogu2617][bzoj1901][Zju2112]Dynamic Rankings【树套树+树状数组+主席树】

    题目网址 [传送门] 题目大意 请你设计一个数据结构,支持单点修改,区间查询排名k. 感想(以下省略脏话inf个字) 真的强力吹爆洛谷数据,一般的树套树还给我T了一般的点,加强的待修主席树还给我卡了几 ...

  8. ZOJ 2112 Dynamic Rankings(树状数组+主席树)

    The Company Dynamic Rankings has developed a new kind of computer that is no longer satisfied with t ...

  9. ZOJ 2112 Dynamic Rankings(树状数组+主席树)

    题意 \(n\) 个数,\(m\) 个操作,每次操作修改某个数,或者询问某个区间的第 \(K\) 小值. \(1 \leq n \leq 50000\) \(1 \leq m \leq 10000\) ...

随机推荐

  1. ASE Alpha Sprint - backend scrum 2

    本次scrum于2019.11.5再sky garden进行,持续30分钟. 参与人: Zhikai Chen, Jia Ning, Haifeng Chen, Hao Wang 请假: Xin Ka ...

  2. JavaScript中准确的判断数据类型

    在 ECMAScript 规范中,共定义了 7 种数据类型,分为基本类型和引用类型两大类. 其中: 基本类型:String.Number.Boolean.Symbol.Undefined.Null  ...

  3. linux强制用户下线命令

    linux强制用户下线命令   前提:必须是root权限操作:(1)使用who查看目前有哪些用户登录了服务器,见下图(2)使用pkill -kill -t pts/1命令踢出第一个用户.命令解释:pt ...

  4. python面向对象--反射机制

    class Black: feture="ugly" def __init__(self,name,addr): self.addr=addr self.name=name def ...

  5. ln 硬链接与软链接

    1. 命令功能 ln 可以看做是link的简写,功能是创建链接文件,链接文件包括硬链接(hard link)和软链接(符号链接,symbolic link) 2. 语法格式 ln  [option]  ...

  6. 新特性2-lambda表达式

    最近几天学习了一下lambda表达式,看了不少博客,感觉有一篇博客总结的一句话总结的很好:lambda表达式是一段可以传递的代码,它的核心思想是将面向对象中的传递数据变成传递行为.其实以前也有传递行为 ...

  7. 多线程、同步实现方法及Error和Exception的区别与联系

    多线程.同步实现方法? 实现线程有两种方法: 继承Thread类 实现Runnable接口 实现同步也有两种方法 一种是用同步方法:同步方法就是在方法返回类型后面加上synchronized, 比如: ...

  8. C#语句,console,C#//,/**/

    ].

  9. redis学习 --List

    1:添加   lpush    rpush     如果没有key的话,会报错.   lpushx    rpushx    如果没有key的话,不做任何反应. 2:得到 lpop   rpop   ...

  10. vim 批量添加删除注释

    vim中单行注释只是多行注释的一个特例,这里统一进行多行注释的讲解 (1)添加批量注释 ctrl+v 进入列编辑模式,向下或向上移动光标,把需要注释的行的开头标记起来,然后按大写的I(shift+i) ...