BZOJ.1901.Dynamic Rankings(线段树套平衡树 Splay)
题意:n个数,有两个操作:1.修改某个数为v;2.询问一段区间第k小的数
如果没有修改,则可以用线段树,每个节点P[a,b]存储大小为b-a+1的数组,代表其中的数
同时,这个数组还是要排好序的
直接找答案很不方便,于是考虑对数组二分答案,求比它小的数的个数
关于构造过程,更新完子节点后,子节点维护的数组就是有序的了,可以通过归并得到父节点的有序数组
这样空间 \(O(nlogn)\),每次查询时间 \(O(log^2n)\)
修改同时有序,二叉排序树是好选择
线段树每个节点维护一棵平衡树,查询时在平衡树中查询;
修改一个点就将它从平衡树中删掉,更改后再插入(所有包含这个点的区间都要执行)
空间复杂度为 \(O(dep*n)\),dep为线段树的深度,大约就是 \(O(nlogn)\)
时间 \(O(nlog^3n)\)
//41644 kb 2404 ms
#include<cstdio>
#include<cctype>
#define gc() getchar()
//#define gc() (SS==TT &&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int N=1e5+5,M=15*N,MAXIN=5e6;
int n,q,tot,A[N],root[N],size,fa[M],son[M][2],sz[M],cnt[M],t[M];
char IN[MAXIN],*SS=IN,*TT=IN;
inline int read()
{
int now=0,f=1;register char c=gc();
for(;!isdigit(c);c=gc()) if(c=='-') f=-1;
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now*f;
}
inline void Update(int rt)
{
sz[rt]=sz[son[rt][0]]+sz[son[rt][1]]+cnt[rt];
}
void Rotate(int x,int &k)
{
int a=fa[x],b=fa[a],l=son[a][1]==x,r=l^1;
if(a==k) k=x;
else son[b][son[b][1]==a]=x;
fa[x]=b, fa[a]=x, fa[son[x][r]]=a,
son[a][l]=son[x][r], son[x][r]=a;
Update(a), Update(x);
}
void Splay(int x,int &k)
{
while(x!=k)
{
int a=fa[x],b=fa[a];
if(a!=k)
{
if((son[a][0]==x)^(son[b][0]==a)) Rotate(x,k);
else Rotate(a,k);
}
Rotate(x,k);
}
}
void Get_Rank(int v,int x)
{
if(!root[x]) return;//!
int k=root[x];
while(t[k]!=v && son[k][v>t[k]]) k=son[k][v>t[k]];
Splay(k,root[x]);
}
void Insert(int v,int x)
{
int f=0,k=root[x];
while(k && v!=t[k]) f=k,k=son[k][v>t[k]];
if(k) ++cnt[k];
else
{
k=++size, cnt[k]=sz[k]=1, t[k]=v, fa[k]=f;
if(f) son[f][v>t[f]]=k;
}
Splay(k,root[x]);
}
void Delete(int v,int x)
{
Get_Rank(v,x);
int k=root[x];
if(cnt[k]>1) {--cnt[k],--sz[k]; return;}
else if(!son[k][0]||!son[k][1]) root[x]=son[k][0]|son[k][1];
else
{
int p=son[k][0];
k=son[k][1], root[x]=k;//!
while(son[k][0]) k=son[k][0];
sz[k]+=sz[p], fa[p]=k, son[k][0]=p;
Splay(k,root[x]);
}
fa[root[x]]=0;//!
}
void Build(int l,int r,int rt,int pos)
{
Insert(A[pos],rt);
if(l==r) return;
int m=l+r>>1;
if(pos<=m) Build(l,m,rt<<1,pos);
else Build(m+1,r,rt<<1|1,pos);
}
void Modify(int l,int r,int rt,int pos,int v)
{
// printf("Modify:%d~%d rt:%d pos:%d v:%d\n",l,r,rt,pos,v);
Delete(A[pos],rt), Insert(v,rt);
if(l>=r) return;
int m=l+r>>1;
if(pos<=m) Modify(l,m,rt<<1,pos,v);
else Modify(m+1,r,rt<<1|1,pos,v);
}
void Calc(int v,int k)
{
while(k)
{
// printf("k:%d v:%d tot:%d\n",k,v,tot);
if(v==t[k]) {tot+=sz[son[k][0]]; return;}
if(v>t[k]) tot+=sz[son[k][0]]+cnt[k],k=son[k][1];
else k=son[k][0];
}
}
void Query(int l,int r,int rt,int L,int R,int v)
{
// printf("Query:%d~%d rt:%d L~R:%d~%d v:%d\n",l,r,rt,L,R,v);
if(L<=l && r<=R) {Calc(v,root[rt]); return;}
int m=l+r>>1;
if(L<=m) Query(l,m,rt<<1,L,R,v);
if(m<R) Query(m+1,r,rt<<1|1,L,R,v);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("1901.in","r",stdin);
// freopen("1901.out","w",stdout);
#endif
n=read(),q=read();
for(int i=1;i<=n;++i) A[i]=read(),Build(1,n,1,i);
char s[5];
int i,j,k;
while(q--)
{
scanf("%s",s),i=read(),j=read();
if(s[0]=='C') Modify(1,n,1,i,j),A[i]=j;
else
{
k=read();
int l=0,r=1e9,m,ans=0;
while(l<=r)
{
m=l+r>>1;
tot=0, Query(1,n,1,i,j,m);
// printf("%d~%d %d\n",l,r,tot);
if(tot>=k) r=m-1;
else l=m+1,ans=m;
}
printf("%d\n",ans);
}
}
return 0;
}
BZOJ.1901.Dynamic Rankings(线段树套平衡树 Splay)的更多相关文章
- [BZOJ 1901] Dynamic Rankings 【树状数组套线段树 || 线段树套线段树】
题目链接:BZOJ - 1901 题目分析 树状数组套线段树或线段树套线段树都可以解决这道题. 第一层是区间,第二层是权值. 空间复杂度和时间复杂度均为 O(n log^2 n). 线段树比树状数组麻 ...
- bzoj 1901 Dynamic Rankings (树状数组套线段树)
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MB Description 给定一个含有n个数的序列a[1] ...
- BZOJ.1901.Dynamic Rankings(树状数组套主席树(动态主席树))
题目链接 BZOJ 洛谷 区间第k小,我们可以想到主席树.然而这是静态的,怎么支持修改? 静态的主席树是利用前缀和+差分来求解的,那么对于每个位置上的每棵树看做一个点,拿树状数组更新. 还是树状数组的 ...
- bzoj 3196/ Tyvj 1730 二逼平衡树 (线段树套平衡树)
3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Description ...
- bzoj 2120 线段树套平衡树
先吐下槽,改了快一个小时,最后发现是SBT的delete写错了,顿时就有想死的心..... 首先对于这道题,我们应该先做一下他的小问题,bzoj1878,虽然和这道题几乎一点关系没有, 但是能给我们一 ...
- 浅谈树套树(线段树套平衡树)&学习笔记
0XFF 前言 *如果本文有不好的地方,请在下方评论区提出,Qiuly感激不尽! 0X1F 这个东西有啥用? 树套树------线段树套平衡树,可以用于解决待修改区间\(K\)大的问题,当然也可以用 ...
- 树套树Day1线段树套平衡树bzoj3196
您需要写一种数据结构,来维护一个有序数列,其中需要提供以下操作:1.查询k在区间内的排名2.查询区间内排名为k的值3.修改某一位值上的数值4.查询k在区间内的前驱(前驱定义为小于x,且最大的数)5.查 ...
- BZOJ.1901.Dynamic Rankings(整体二分)
题目链接 BZOJ 洛谷 (以下是口胡) 对于多组的询问.修改,我们可以发现: 假设有对p1,p2,p3...的询问,在这之前有对p0的修改(比如+1),且p0<=p1,p2,p3...,那么我 ...
- 【BZOJ-3196】二逼平衡树 线段树 + Splay (线段树套平衡树)
3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2271 Solved: 935[Submit][Stat ...
随机推荐
- Protobuf使用手册
Protobuf使用手册 第1章 定义.proto 文件 首先我们需要编写一个 proto 文件,定义我们程序中需要处理的结构化数据,在 protobuf 的术语中,结构化数据被称为 Message. ...
- freeRTOS中文实用教程3--中断管理之延迟中断处理
1.前言 嵌入式实时操作系统需要对整个系统环境产生的事件作出响应.可以采用中断方式也可以采用轮询方式来进行处理.如果采用中断方式,则希望ISR(中断服务例程)的处理时间越短越好. 注:必须说明的是,只 ...
- kindle转换工具-calibre
kindle转换工具 calibre https://calibre-ebook.com/download_windows
- 使用C#进行应用程序间通信(WPF与Unity通信)
首先程序主体来自网络,我只是应用在我自己的项目中,其中出现了一系列的问题,有些已经解决,有些使用了折中的方案,如果有大神能够给予知道,感激不尽! 首先是发送端程序: 这是我的程序任务执行主界面,此处已 ...
- day1 diff命令递归比较目录下的文件
一.diff实战 (1)递归比较文件夹下所有的文件及目录的不同 diff --brief -Nr dir1/ dir2/ Reference ...
- insmod 时报错“Unknown symbol”问题的解决
在加载驱动模块时报错: “ Unknown symbol CFG80211_SupBandReInit (err 0)” 查看了内核代码以及加载上的symbol(命令为 cat /proc/kalls ...
- activiti报错ProcessEngines.getDefaultProcessEngine()为null
activiti报错ProcessEngines.getDefaultProcessEngine()为null 文件名错误,默认加载classpath下面的activiti.cfg.xml,而不是ac ...
- html dom之iframe对象
当从父页面中需要获取使用iframe嵌入的内容时,可以使用图中后面的两个属性 var sonDocument = document.getElementById('iframe_id').conten ...
- ThinkPHP 3.1,3.2中对IN和BETWEEN正则匹配不当导致的一个SQLi
// where子单元分析 protected function parseWhereItem($key,$val) { $whereStr = ''; if(is_array($val)) { if ...
- TomCat安装配置教程
一.JDK的安装与配置 1.从官网下载jdk,注意是jdk不是jre.最好从官网下载,也可以直接度娘. 2.下载完毕后,安装jdk,直接按照安装向导的提示安装即可,安装时可以自己选择安装路径,我的安 ...