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 ...
随机推荐
- 【黑客免杀攻防】读书笔记7 - 软件逆向工程基础1(函数调用约定、Main函数查找)
0x1 准备工作 1.1.准备工具 IDA:交互式反汇编工具 OllyDbg:用户层调试工具 Visual Studio:微软开发工具 1.2.基础知识 C++开发 汇编语言 0x2 查找真正的mai ...
- 关于Mybatis的SQL映射文件中in关键字的用法
有一个需求是可以选择多个设备进行删除,于是想到将多个设备id拼成字符串作为参数,以逗号隔开,如:"123,234,456". SQL如下: <delete id=" ...
- softlab对接Jenkins工程
在Softlab上对接jenkins工程,必须先在自己的jenkins环境上配置好任务. 1.登陆到jenkins首页,比如http://10.179.175.53:8080/jenkins/ 2.点 ...
- Android动画分类
动画分类 View动画(补间动画).帧动画.属性动画 View动画(补间动画)包括:平移.旋转.缩放.透明度,View动画是一种渐近式动画 帧动画:图片切换动画 属性动画:通过动态改变对象的属性达到动 ...
- python抓取bing主页背景图片
最初Python2写法: #!/usr/bin/env python # -*- coding:utf-8 -*- # -*- author:nancy -*- # python2抓取bing主页所有 ...
- window 命令大全
运行操作 CMD命令:开始->运行->键入cmd或command(在命令行里可以看到系统版本.文件系统版本) CMD命令锦集 1. gpedit.msc-----组策略 2. ...
- 在 laravel 的 DB::transaction 中,为外部变量赋值
例如,我想在 laravel 的事务中,对某个外部变量赋值,然后在后续的逻辑中判断该变量的属性 $user = null; // init DB::transaction(function() use ...
- SPLAY,LCT学习笔记(一)
写了两周数据结构,感觉要死掉了,赶紧总结一下,要不都没学明白. SPLAY专题: 例:NOI2005 维修数列 典型的SPLAY问题,而且综合了SPLAY常见的所有操作,特别适合新手入门学习(比如我这 ...
- linux安装memcached安装以及memcache的php扩展
https://blog.csdn.net/sshcnwold/article/details/79085082
- jQuery插件实践之轮播练习(一)
所有文章搬运自我的个人主页:sheilasun.me 因为从来没写过jQuery插件,所以本文要通过一个轮播的例子,练习jQuery插件的写法. 新建插件文件 在讨论细节之前,先新建插件文件(当然也可 ...