本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作。

本文作者:ljh2000
作者博客:http://www.cnblogs.com/ljh2000-jump/
转载请注明出处,侵权必究,保留最终解释权!

题目链接:BZOJ3224

正解:$Treap$

解题报告:

  $Treap$其实就是二叉搜索树啦,加入了随机化权值之后可以保证树的形态比较平衡。

  $Treap$既是一棵二叉搜索树又是一个大根堆(小根堆也行,随意啦),每次$insert$了之后给这个节点$rand$一个权值就好了,然后每次需要保证堆的性质,所以要维护一个向上旋转的操作。

  这就是旋转式的$Treap$了,挺simple的。

//It is made by ljh2000
//有志者,事竟成,破釜沉舟,百二秦关终属楚;苦心人,天不负,卧薪尝胆,三千越甲可吞吴。
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <ctime>
#include <vector>
#include <queue>
#include <map>
#include <set>
#include <string>
#include <complex>
#include <bitset>
using namespace std;
typedef long long LL;
typedef long double LB;
typedef complex<double> C;
const double pi = acos(-1);
const int MAXN = 100011;
int n,rt,tot,tr[MAXN][2],size[MAXN],r[MAXN],val[MAXN];
//维护按权值的二叉搜索树,同时是随机权值的大根堆
//相等的均靠左
inline void update(int x){ size[x]=size[tr[x][0]]+size[tr[x][1]]+1; }
inline int getint(){
int w=0,q=0; char c=getchar(); while((c<'0'||c>'9') && c!='-') c=getchar();
if(c=='-') q=1,c=getchar(); while (c>='0'&&c<='9') w=w*10+c-'0',c=getchar(); return q?-w:w;
} inline void R(int &p,bool k){//把x的左/右儿子旋转上来
int t=tr[p][k]; tr[p][k]=tr[t][k^1]; tr[t][k^1]=p;
update(p); update(t); p=t;
} inline void insert(int &p,int x){
if(!p) { p=++tot; val[p]=x; size[p]=1; r[p]=rand(); return ; }
size[p]++;//!!!
if(x<=val[p]) insert(tr[p][0],x); else insert(tr[p][1],x);
bool k=(x>val[p]);
if(r[ tr[p][k] ] > r[p]) R(p,k);//不满足大根堆性质,往上旋转
} inline void out(int &p){//删除当前节点,不断旋转,直到没有左儿子或者没有右儿子就可以结束了
if(!tr[p][0] || !tr[p][1]) { p=tr[p][0]+tr[p][1]; return ; }
bool k=(r[ tr[p][1] ]>r[ tr[p][0] ]);//把key偏大的那个儿子旋转上来
R(p,k); size[p]--;//!!!
out(tr[p][k^1]);
} inline void del(int &p,int k){//把当前子树中rank为k的结点删除
size[p]--;//!!!在最前面呀...
if(size[tr[p][0]]+1==k) { out(p); return ; }
if(size[tr[p][0]]>=k) del(tr[p][0],k);
else del(tr[p][1]/*!!!*/,k-size[tr[p][0]]-1);
} inline int kth(int x,int k){//查询rank为k的数,rank最小的那个
while(x) {
if(size[tr[x][0]]+1==k) return val[x];
if(k<=size[tr[x][0]]) x=tr[x][0];
else k-=size[tr[x][0]]+1,x=tr[x][1];
}
return 0;
} inline int rank(int x,int k){//查询>=k的第一个数的rank,最靠左的那个
int tot=1;
while(x) {
if(k<=val[x]) x=tr[x][0];
else tot+=size[tr[x][0]]+1,x=tr[x][1];
}
return tot;
} inline void work(){
srand(23333);
n=getint(); int type,x;
while(n--) {
type=getint(); x=getint();
if(type==1) {
insert(rt,x);
}
else if(type==2) {
del(rt,rank(rt,x));
}
else if(type==3) {
printf("%d\n",rank(rt,x));
}
else if(type==4) {
printf("%d\n",kth(rt,x));
}
else if(type==5) {
printf("%d\n", kth( rt, rank(rt,x)-1 ) );
}
else {
printf("%d\n", kth( rt, rank(rt,x+1) ) );
}
}
} int main()
{
#ifndef ONLINE_JUDGE
freopen("3224.in","r",stdin);
freopen("3224.out","w",stdout);
#endif
work();
return 0;
}
//有志者,事竟成,破釜沉舟,百二秦关终属楚;苦心人,天不负,卧薪尝胆,三千越甲可吞吴。

  

BZOJ3224 Tyvj 1728 普通平衡树(Treap)的更多相关文章

  1. [BZOJ3224] [Tyvj 1728] 普通平衡树 (treap)

    Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作: 1. 插入x数 2. 删除x数(若有多个相同的数,因只删除一个) 3. 查询x数的排名(若有多个相 ...

  2. [BZOJ3224]Tyvj 1728 普通平衡树

    [BZOJ3224]Tyvj 1728 普通平衡树 试题描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个) ...

  3. bzoj3224: Tyvj 1728 普通平衡树(平衡树)

    bzoj3224: Tyvj 1728 普通平衡树(平衡树) 总结 a. cout<<(x=3)<<endl;这句话输出的值是3,那么对应的,在splay操作中,当父亲不为0的 ...

  4. BZOJ 3224 TYVJ 1728 普通平衡树 [Treap树模板]

    3224: Tyvj 1728 普通平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB Submit: 7390  Solved: 3122 [Submit][S ...

  5. bzoj3224 Tyvj 1728 普通平衡树(名次树+处理相同)

    3224: Tyvj 1728 普通平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 5354  Solved: 2196[Submit][Sta ...

  6. BZOJ 3224: Tyvj 1728 普通平衡树 treap

    3224: Tyvj 1728 普通平衡树 Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除 ...

  7. bzoj3224: Tyvj 1728 普通平衡树(splay)

    3224: Tyvj 1728 普通平衡树 题目:传送门 题解: 啦啦啦啦又来敲个模版水经验啦~ 代码: #include<cstdio> #include<cstring> ...

  8. 【权值线段树】bzoj3224 Tyvj 1728 普通平衡树

    一个板子. #include<cstdio> #include<algorithm> using namespace std; #define N 100001 struct ...

  9. BZOJ3224——Tyvj 1728 普通平衡树

    1.题目大意:数据结构题,是treap,全都是treap比较基本的操作 2.分析:没啥思考的 #include <cstdio> #include <cstdlib> #inc ...

随机推荐

  1. git提交到远程虚拟机

    git到自己的虚拟机中第一步:打通git(一)Linux中(ip为10.1.8.1)1.安装git如:Ubuntu中安装gitapt install git 2.Ubuntu中添加git用户sudo ...

  2. PAT 1096 Consecutive Factors[难]

    1096 Consecutive Factors (20 分) Among all the factors of a positive integer N, there may exist sever ...

  3. cnn for qa

    最近在做QA系统,用tensorflow做了些实验,下面的的是一个cnn的评分网络.主要参考了<APPLYING DEEP LEARNING TO ANSWER SELECTION: A STU ...

  4. python字符串的操作(去掉空格strip(),切片,查找,连接join(),分割split(),转换首字母大写, 转换字母大小写...)

    #可变变量:list, 字典#不可变变量:元祖,字符串字符串的操作(去掉空格, 切片, 查找, 连接, 分割, 转换首字母大写, 转换字母大小写, 判断是否是数字字母, 成员运算符(in / not ...

  5. adb push ,adb pull和adb install的区别

    1.用命令行把手机上的文件拷贝到电脑上 1 adb pull sdcard/1222073679.png 拷贝文件夹命令,如把log文件夹拷贝到电脑当前目录 1 adb pull sdcard/log ...

  6. Linux系统——文件系统与LVM 逻辑

    格式化命令 mkfs. mkswap mkfs格式化数据磁盘 # mkfs -t ext4 /dev/sdb1 # mkfs.ext4 /dev/sdb1 -t 指定格式化文件类型 -b 指定bloc ...

  7. mellanox RDMA RoCE

    一:首先根据系统发行版本下载对应的驱动,下载地址如下: http://www.mellanox.com/page/products_dyn?product_family=26&mtag=lin ...

  8. JS知识点简单总结

    JS(JavaScript) 是一种解释性语言 是弱/动态 最开始用于表单验证的 JS的五个部分: 1.核心语言定义 2.原生对象和内置对象 5boolean 3.BOM 4.DOM 5.事件处理模型 ...

  9. Java对map进行排序并生成序号

    最近做的项目有这样一个需求:要求对map中的值进行排序并生成序号.如果值相等则序号不变:如果不相等序号为该数数值在所有元素中的索引.如下表所示: Key(String) Value(Float) Id ...

  10. 关于JavaScript对象,你所不知道的事(一)- 先谈对象

    这篇博文的主要目的是为了填坑,很久之前我发表了一篇名为关于JavaScript对象中的一切(一) - 对象属性的文章,想要谈一谈JavaScript对象,可那时只是贴了一张关于这个主题的思维导图,今天 ...