比较裸的树套树,对于区间K值bz上有一道裸题,详见题解http://www.cnblogs.com/BLADEVIL/p/3455336.html(其实题解也不是很详细)

//By BLADEVIL
type
rec =record
left, right, root :longint;
end; var
n, m :longint;
a :array[..] of longint;
t :array[..] of rec;
b_left, b_right, b_key :array[..] of longint;
b_size :array[..] of longint;
tot :longint; function min(a,b:longint):longint;
begin
if a>b then min:=b else min:=a;
end; function max(a,b:longint):longint;
begin
if a>b then max:=a else max:=b;
end; procedure left_rotate(var t:longint);
var
k :longint;
begin
k:=b_right[t];
b_right[t]:=b_left[k];
b_left[k]:=t;
b_size[k]:=b_size[t];
b_size[t]:=b_size[b_left[t]]+b_size[b_right[t]]+;
t:=k;
end; procedure right_rotate(var t:longint);
var
k :longint;
begin
k:=b_left[t];
b_left[t]:=b_right[k];
b_right[k]:=t;
b_size[k]:=b_size[t];
b_size[t]:=b_size[b_left[t]]+b_size[b_right[t]]+;
t:=k;
end; procedure maintain(var t:longint;flag:boolean);
begin
if not flag then
begin
if b_size[b_left[b_left[t]]]>b_size[b_right[t]] then
right_rotate(t) else
if b_size[b_right[b_left[t]]]>b_size[b_right[t]] then
begin
left_rotate(b_left[t]);
right_rotate(t);
end else exit;
end else
begin
if b_size[b_right[b_right[t]]]>b_size[b_left[t]] then
left_rotate(t) else
if b_size[b_left[b_right[t]]]>b_size[b_left[t]] then
begin
right_rotate(b_right[t]);
left_rotate(t);
end else exit;
end;
maintain(b_left[t],false);
maintain(b_right[t],true);
maintain(t,true);
maintain(t,false);
end; procedure insert(var t:longint;v:longint);
begin
if t= then
begin
inc(tot);
t:=tot;
b_left[t]:=;
b_right[t]:=;
b_size[t]:=;
b_key[t]:=v;
end else
begin
inc(b_size[t]);
if v>=b_key[t] then insert(b_right[t],v) else insert(b_left[t],v);
maintain(t,v>=b_key[t]);
end;
end; function delete(var t:longint; v:longint):longint;
begin
dec(b_size[t]);
if (b_key[t]=v) or (b_key[t]<v) and (b_right[t]=) or (b_key[t]>v) and (b_left[t]=) then
begin
delete:=b_key[t];
if (b_left[t]=) or (b_right[t]=) then
t:=b_left[t]+b_right[t] else
b_key[t]:=delete(b_left[t],v+);
end else
if v>b_key[t] then delete:=delete(b_right[t],v) else delete:=delete(b_left[t],v);
end; procedure build(x,l,r:longint);
var
mid :longint;
i :longint;
begin
t[x].left:=l; t[x].right:=r; t[x].root:=;
for i:=l to r do insert(t[x].root,a[i]);
if l=r then exit;
with t[x] do mid:=(left+right) div ;
build(x*,l,mid); build(x*+,mid+,r);
end; function b_rank(var t:longint;v:longint):longint;
begin
if t= then exit();
if v<=b_key[t] then
b_rank:=b_rank(b_left[t],v) else
b_rank:=b_rank(b_right[t],v)+b_size[b_left[t]]+;
end; procedure init;
var
i :longint; begin
read(n,m);
for i:= to n do read(a[i]);
build(,,n);
end; function rank(x,l,r,y:longint):longint;
var
mid :longint;
begin
if (t[x].left=l) and (t[x].right=r) then
begin
rank:=b_rank(t[x].root,y);
exit;
end;
with t[x] do mid:=(left+right) div ;
if mid<l then rank:=rank(x*+,l,r,y) else
if mid>=r then rank:=rank(x*,l,r,y) else
rank:=rank(x*,l,mid,y)+rank(x*+,mid+,r,y);
end; procedure askrank(l,r,x:longint);
begin
writeln(rank(,l,r,x)+);
end; procedure change(x,y,z,c:longint);
var
mid :longint;
begin
mid:=delete(t[x].root,c);
insert(t[x].root,z);
if t[x].left=t[x].right then exit;
with t[x] do mid:=(left+right) div ;
if y>mid then change(x*+,y,z,c) else
change(x*,y,z,c);
end; function b_pred(var t:longint;v:longint):longint;
begin
if t= then exit(-);
if v<=b_key[t] then b_pred:=b_pred(b_left[t],v) else
begin
b_pred:=b_pred(b_right[t],v);
if b_pred=- then b_pred:=b_key[t];
end;
end; function pred(x,l,r,y:longint):longint;
var
mid :longint;
begin
if (t[x].left=l) and (t[x].right=r) then
begin
pred:=b_pred(t[x].root,y);
exit;
end;
with t[x] do mid:=(left+right) div ;
if mid<l then pred:=pred(x*+,l,r,y) else
if mid>=r then pred:=pred(x*,l,r,y) else
pred:=max(pred(x*,l,mid,y),pred(x*+,mid+,r,y));
end; procedure askpred(l,r,x:longint);
begin
writeln(pred(,l,r,x));
end; function b_succ(var t:longint;v:longint):longint;
begin
if t= then exit(maxlongint);
if v>=b_key[t] then b_succ:=b_succ(b_right[t],v) else
begin
b_succ:=b_succ(b_left[t],v);
if b_succ=maxlongint then b_succ:=b_key[t];
end;
end; function succ(x,l,r,y:longint):longint;
var
mid :longint;
begin
if (t[x].left=l) and (t[x].right=r) then
begin
succ:=b_succ(t[x].root,y);
exit;
end;
with t[x] do mid:=(left+right) div ;
if mid<l then succ:=succ(x*+,l,r,y) else
if mid>=r then succ:=succ(x*,l,r,y) else
succ:=min(succ(x*,l,mid,y),succ(x*+,mid+,r,y));
end; procedure askselect(a,b,x:longint);
var
mid :longint;
ans :longint;
l, r :longint;
xx :longint;
begin
l:=; r:=;
while l<=r do
begin
mid:=(l+r) div ;
xx:=rank(,a,b,mid)+;
if xx<=x then
begin
ans:=mid;
l:=mid+;
end else r:=mid-;
end;
if rank(,a,b,ans)=x then ans:=succ(,l,r,ans-) else ans:=ans;
writeln(ans);
end; procedure asksucc(l,r,x:longint);
begin
writeln(succ(,l,r,x));
end; procedure main;
var
i :longint;
k, l, r, x :longint; begin
for i:= to m do
begin
read(k);
if k= then
begin
read(l,r,x);
askrank(l,r,x);
end else
if k= then
begin
read(l,r,x);
askselect(l,r,x);
end else
if k= then
begin
read(l,x);
change(,l,x,a[l]);
a[l]:=x;
end else
if k= then
begin
read(l,r,x);
askpred(l,r,x);
end else
if k= then
begin
read(l,r,x);
asksucc(l,r,x);
end;
end;
end; begin
init;
main;
end.

bzoj 3196二逼平衡树 线段树套平衡树的更多相关文章

  1. BZOJ3196二逼平衡树——线段树套平衡树(treap)

    此为平衡树系列最后一道:二逼平衡树您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名2.查询区间内排名为k的值3.修改某一位值上的数值4.查询 ...

  2. bzoj3196 二逼平衡树——线段树套平衡树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3196 人生中第一棵树套树! 写了一个晚上,成功卡时 9000ms+ 过了! 很要注意数组的大 ...

  3. [bzoj3196]Tyvj 1730 二逼平衡树——线段树套平衡树

    题目 Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的数值 4.查 ...

  4. P3380 【模板】二逼平衡树(树套树)(线段树套平衡树)

    P3380 [模板]二逼平衡树(树套树) 前置芝士 P3369 [模板]普通平衡树 线段树套平衡树 这里写的是线段树+splay(不吸氧竟然卡过了) 对线段树的每个节点都维护一颗平衡树 每次把给定区间 ...

  5. bzoj 3196/ Tyvj 1730 二逼平衡树 (线段树套平衡树)

    3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description ...

  6. 【BZOJ-3196】二逼平衡树 线段树 + Splay (线段树套平衡树)

    3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2271  Solved: 935[Submit][Stat ...

  7. P3380 【模板】二逼平衡树(树套树) 线段树套平衡树

    \(\color{#0066ff}{ 题目描述 }\) 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 查询k在区间内的排名 查询区间内排名为k的值 修改某一位值上 ...

  8. BZOJ3196 二逼平衡树 【线段树套平衡树】

    题目 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的数值 4.查询k在区间内的前驱(前驱 ...

  9. BZOJ 3196 二逼平衡树

    Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:1.查询k在区间内的排名2.查询区间内排名为k的值3.修改某一位值上的数值4.查询k在区间内的 ...

随机推荐

  1. Anytime项目开发记录0

    Anytime,中文名:我很忙. 开发者:孤独的猫咪神. 这个项目会持续更新,直到我决定不再维护这个APP. 2014年3月10日:近日有事,暂时断更.希望可以会尽快完事. 2014年3月27日:很抱 ...

  2. Linux下创建pycharm的快捷方式

    第一步:创建桌面快捷方式文件Pycharm.desktop,并打开 sudo gedit /usr/share/applications/Pycharm.desktop 第二步:在打开的文件Pycha ...

  3. tinymce4.x 上传本地图片 (转载)

    转载自:http://www.cnblogs.com/fhen/p/5809514.html tinymce4.x 上传本地图片   tinymce是一款挺不错的html文本编辑器.但是添加图片是直接 ...

  4. 简易cmake多文件多目录工程模板

    今天心血来潮,想在服务器上试试写libevent的工程是什么感受,那第一步就是学会怎么用cmake建工程,之前也没接触过cmake,然后一上午,比较懵逼,下午看实验室哥们给的一个教程,然后,慢慢理解C ...

  5. HDU 4467 Graph(图论+暴力)(2012 Asia Chengdu Regional Contest)

    Description P. T. Tigris is a student currently studying graph theory. One day, when he was studying ...

  6. javascript知识总结

    javascript: 面对对象 函数创建方式: 1.工厂模式 function createPerson(name, age, job){ var o = new Object(); //创建工厂对 ...

  7. Anaconda使用入门

    简介 Conda是一个开源的包.环境管理器,可以用于在同一个机器上安装不同版本的软件包及其依赖,并能够在不同的环境之间切换 Anaconda包括Conda.Python以及一大堆安装好的工具包,比如: ...

  8. JS容易理解错误的地方

    在这端代码执行的末尾,你会不会hi变量回事函数中的hi了?你会不会认为这不是按引用传递了? 对值传递和引用传递产生质疑了? 1 var hi = {}; 2 function sayHello(hi) ...

  9. [CERC2017]Intrinsic Interval——扫描线+转化思想+线段树

    [CERC2017]Intrinsic Interval https://www.luogu.org/blog/ywycasm/solution-p4747# 这种“好的区间”,见得还是比较多的了. ...

  10. [学习笔记]LCT进阶操作

    LCT总结——应用篇(附题单)(LCT) 一般都是维护链的操作.split即可搞定. 进阶操作的话,处理好辅助树和原树的关系即可搞定. 其实,最大的区别就是,splay随便转,辅助树形态变了,但是原树 ...