BZOJ1901 Zju2112 Dynamic Rankings 主席树
欢迎访问~原文出处——博客园-zhouzhendong
去博客园看该题解
题目传送门 - BZOJ1901
题意概括
给你一段序列(n个数),让你支持一些操作(共m次),
有两种操作,一种是询问区间第k小,一种是单点修改。
n,m<=10000
题解
这个主席树的写法是我自己造出来的。
主席树的查询区间第k大需要依赖前缀和。
树状数组最擅长这个了,就让他来干。
原理是这样的:
先离散化,包括修改操作里面的数字也要离散化。
然后建树,包括修改操作所涉及的数值也要建。
现在总共有n个线段树。
第0棵树是最完整的,故我们把这棵树的每一个节点都当作一个树状数组的起点,即该树状数组的第一项。
然后往下一棵树走,当然下一棵树不一定有这个节点,那么不停的走下去,这个可以在建树的时候预先处理一个Next数组来优化。
走到下一个这个位置的节点之后,我们再把这个节点作为该树状数组的第二项。
这样我们有大约n*4(第0棵树的节点个数)个树状数组了。
然而不处理要超空间。
所以有两种方案,一种是vector,一种是把所有的树状数组连起来,变成一个大的。
然而查询之类的就和普通的主席树差不多了。
细节上要大大的注意!
代码
#include <cstring>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <vector>
using namespace std;
const int N=10005,S=N*2*20*4;//v=4*node=4*20*hs=4*20*2*n
struct Que{
char op;
int a,b,c;
void read(){
char opc[5];
scanf("%s%d%d",opc,&a,&b);
op=opc[0];
if (op=='Q')
scanf("%d",&c);
}
}q[N];
vector <int> vec[N];
int n,m,v[N],Ha[N*2],hs=0,total=0,totarr=0;
int root[N],ls[S],rs[S],sum[S],pos[S],Next[S],p[S];
void LSH(){
int hs_=1;
sort(Ha+1,Ha+hs+1);
for (int i=2;i<=hs;i++)
if (Ha[i]!=Ha[i-1])
Ha[++hs_]=Ha[i];
hs=hs_;
}
int find(int x){
int L=1,R=hs,mid;
while (L<=R){
mid=(L+R)>>1;
if (Ha[mid]==x)
return mid;
if (Ha[mid]<x)
L=mid+1;
else
R=mid-1;
}
exit(233);
}
int build(int L,int R){
int rt=++total;
if (L==R){
ls[rt]=rs[rt]=0;
return rt;
}
int mid=(L+R)>>1;
ls[rt]=build(L,mid);
rs[rt]=build(mid+1,R);
return rt;
}
int lowbit(int x){
return x&-x;
}
void add(int x,int v){
for (;x<=totarr;x+=lowbit(x))
sum[x]+=v;
}
int Sum(int x){
int ans=0;
for (;x>0;x-=lowbit(x))
ans+=sum[x];
return ans;
}
void add_tree(int prt,int &rt,int L,int R,int pos){
if (!rt||rt==prt)
rt=++total;
Next[prt]=rt;
if (L==R)
return;
int mid=(L+R)>>1;
if (pos<=mid){
add_tree(ls[prt],ls[rt],L,mid,pos);
if (!rs[rt])
rs[rt]=rs[prt];
}
else {
add_tree(rs[prt],rs[rt],mid+1,R,pos);
if (!ls[rt])
ls[rt]=ls[prt];
}
}
void build_treearr(int srt,int L,int R){
for (int i=srt;i;i=Next[i])
p[i]=++totarr;
if (L==R)
return;
int mid=(L+R)>>1;
build_treearr(ls[srt],L,mid);
build_treearr(rs[srt],mid+1,R);
}
void update(int rt,int L,int R,int pos,int v){
add(p[rt],v);
if (L==R)
return;
int mid=(L+R)>>1;
if (pos<=mid)
update(ls[rt],L,mid,pos,v);
else
update(rs[rt],mid+1,R,pos,v);
}
int query(int prt,int rt,int L,int R,int k){
if (L==R)
return Ha[L];
int mid=(L+R)>>1;
int LV=Sum(p[ls[rt]])-Sum(p[ls[prt]]);
if (k<=LV)
return query(ls[prt],ls[rt],L,mid,k);
else
return query(rs[prt],rs[rt],mid+1,R,k-LV);
}
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++)
scanf("%d",&v[i]),Ha[++hs]=v[i];
for (int i=1;i<=m;i++){
q[i].read();
if (q[i].op=='C')
Ha[++hs]=q[i].b;
}
LSH();
for (int i=1;i<=n;i++)
vec[i].clear();
for (int i=1;i<=n;i++)
vec[i].push_back(v[i]);
for (int i=1;i<=m;i++)
if (q[i].op=='C')
vec[q[i].a].push_back(q[i].b);
memset(sum,0,sizeof sum);
memset(pos,0,sizeof pos);
memset(Next,0,sizeof Next);
memset(root,0,sizeof root);
root[0]=build(1,hs);
for (int i=1;i<=n;i++)
for (int j=0;j<vec[i].size();j++)
add_tree(root[i-1],root[i],1,hs,find(vec[i][j]));
build_treearr(root[0],1,hs);
for (int i=1;i<=n;i++)
update(root[i],1,hs,find(v[i]),1);
for (int i=1;i<=m;i++)
if (q[i].op=='C'){
update(root[q[i].a],1,hs,find(v[q[i].a]),-1);
update(root[q[i].a],1,hs,find(v[q[i].a]=q[i].b),1);
}
else
printf("%d\n",query(root[q[i].a-1],root[q[i].b],1,hs,q[i].c));
return 0;
}
BZOJ1901 Zju2112 Dynamic Rankings 主席树的更多相关文章
- 【BZOJ1901】Zju2112 Dynamic Rankings 主席树+树状数组
[BZOJ1901]Zju2112 Dynamic Rankings Description 给定一个含有n个数的序列a[1],a[2],a[3]……a[n],程序必须回答这样的询问:对于给定的i,j ...
- Bzoj 1901: Zju2112 Dynamic Rankings 主席树,可持久,树状数组,离散化
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 6321 Solved: 2628[Su ...
- bzoj 1901: Zju2112 Dynamic Rankings -- 主席树,树状数组,哈希
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MB Description 给定一个含有n个数的序列a[1] ...
- [luogu2617][bzoj1901][Zju2112]Dynamic Rankings【树套树+树状数组+主席树】
题目网址 [传送门] 题目大意 请你设计一个数据结构,支持单点修改,区间查询排名k. 感想(以下省略脏话inf个字) 真的强力吹爆洛谷数据,一般的树套树还给我T了一般的点,加强的待修主席树还给我卡了几 ...
- BZOJ1901 Zju2112 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- ...
- [bzoj1901][Zju2112]Dynamic Rankings_主席树
Dynamic Rankings bzoj-1901 Zju-2112 题目大意:给定一个n个数的序列,m个操作,支持:单点修改:查询区间k小值. 注释:$1\le n,m\le 10^4$. 想法: ...
- [BZOJ1901]Zju2112 Dynamic Rankings
[BZOJ1901]Zju2112 Dynamic Rankings 试题描述 给定一个含有n个数的序列a[1],a[2],a[3]--a[n],程序必须回答这样的询问:对于给定的i,j,k,在a[i ...
- BZOJ-1901 Zju2112 Dynamic Rankings 函数式线段树 套 树状数组+离线处理
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MB Submit: 6058 Solved: 2521 [Su ...
- bzoj1901: Zju2112 Dynamic Rankings(BIT套主席树)
带修改的题主席树不记录前缀,只记录单点,用BIT统计前缀. 对于BIT上每一个点建一棵主席树,修改和询问的时候用BIT跑,在主席树上做就行了. 3k4人AC的题#256...应该不算慢 #incl ...
随机推荐
- VUE优秀的组件库总结
VUE组件库 vux github ui demo:https://github.com/airyland/vux Mint UI 项目主页:http://mint-ui.github.io/#!/z ...
- static, const
static 静态的,类的静态成员函数,静态成员变量是和类相关的,但不和具体对象相关.即使没有具体对象,也能调用类的静态成员函数和成员变量.一般类的静态函数就是一个全局函数,只是作用域在包含它的文件中 ...
- C++内存管理(转)http://www.cnblogs.com/qiubole/archive/2008/03/07/1094770.html
[导语] 内存管理是C++最令人切齿痛恨的问题,也是C++最有争议的问题,C++高手从中获得了更好的性能,更大的自由,C++菜鸟的收获则是一遍一遍的检查代码和对C++的痛恨,但内存管理在C++中无处不 ...
- saltstack系列~第二篇
一 简介:今天咱们来继续学习saltstack 二 命名和分组 1 命名规则 1 ID构成 机房-DB类型-角色(主/从)-IP地址 2 分组构成 分为master slave两组即可 2 分组规则 ...
- 运维数据库平台~inception审核规则详解
---恢复内容开始--- 一 简介:这次我们来介绍最核心的审核功能 二 讲解:简单来说 inception就是mysql的二次过滤,何谓二次过滤,我们知道,mysql本身都有自己的审核规则,为业界所通 ...
- 如何调整cell的大小
一般情况下,我们使用tableview的时候从来没有设置过cell的大小(w,h).位置(x,y)等,而是系统直接给我们自动生成,但是有的时候我们可能会改动cell的大小及位置,比如:在适配ios6跟 ...
- block循环引用
block里边会有循环引用的风险,它可能对外部一个变量出现强引用,所以需要判断里边是否有循环引用,通过dealloc方法(销毁当前控制器.或销毁要测试的变量),判断是否循环引用.主要在block 里边 ...
- Linux下clock计时函数学习
平时在Linux和Winows下都有编码的时候,移植代码的时候免不了发现一些问题.1. 你到底准不准?关于clock()计时函数首先是一段简单的测试代码,功能为测试从文本文件读取数据并赋值给向量最后打 ...
- linux统计某个特定文件名的大小总和【原创】
[hch@EAISRVBJ2 log]$find ./ -name "test_chs_00*"|xargs du -ck|grep total|awk 'BEGIN{sum=0} ...
- python高级编程读书笔记(一)
python高级编程读书笔记(一) python 高级编程读书笔记,记录一下基础和高级用法 python2和python3兼容处理 使用sys模块使程序python2和python3兼容 import ...