Hihocoder 1325 平衡树·Treap(平衡树,Treap)

Description

小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似。

小Hi:你说的是哪两个啊?

小Ho:就是二叉排序树和堆啊,你看这两种数据结构都是构造了一个二叉树,一个节点有一个父亲和两个儿子。 如果用1..n的数组来存储的话,对于二叉树上的一个编号为k的节点,其父亲节点刚好是k/2。并且它的两个儿子节点分别为k2和k2+1,计算起来非常方便呢。

小Hi:没错,但是小Hi你知道有一种办法可以把堆和二叉搜索树合并起来,成为一个新的数据结构么?

小Ho:这我倒没想过。不过二叉搜索树满足左子树<根节点<右子树,而堆是满足根节点小于等于(或大于等于)左右儿子。这两种性质是冲突的啊?

小Hi:恩,你说的没错,这两种性质的确是冲突的。

小Ho:那你说的合并是怎么做到的?

小Hi:当然有办法了,其实它是这样的....

Input

第1行:1个正整数n,表示操作数量,10≤n≤100,000

第2..n+1行:每行1个字母c和1个整数k:

若c为'I',表示插入一个数字k到树中,-1,000,000,000≤k≤1,000,000,000

若c为'Q',表示询问树中不超过k的最大数字

Output

若干行:每行1个整数,表示针对询问的回答,保证一定有合法的解

Sample Input

5

I 3

I 2

Q 3

I 5

Q 4

Sample Output

3

3

Http

Hihocoder:http://hihocoder.com/problemset/problem/1325?sid=1122544

Source

二叉平衡树 Treap

解决思路

Treap学习题(待以后补充)

代码

/*
警告:本题代码或许在指针使用上存在问题,等待博主重构数组版。(虽然说这份代码能在Hihocoder上通过,但有读者反映可能会出现RE)
*/
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std; class Treap
{
public:
int key,ran;
int size;
Treap * ch[2];
Treap(int k)
{
key=k;
ran=rand();
ch[0]=ch[1]=NULL;
size=1;
}
int compare(int k)
{
if (k==key)
return -1;
return k<key?0:1;
}
void maintain()
{
size=1;
if (ch[0]!=NULL)
size+=ch[0]->size;
if (ch[1]!=NULL)
size+=ch[1]->size;
}
}; const int inf=2147483647; int n; void Rotate(Treap* &T,int f);//0代表左旋,1代表右旋
void Insert(Treap* &T,int value);
int Find(Treap * T,int value);
int Find_k(Treap * T,int value);
void print(Treap *T); int main()
{
Treap* root=NULL;
cin>>n;
for (int i=1;i<=n;i++)
{
char ch;
int x;
cin>>ch>>x;
if (ch=='I')
{
//cout<<"x "<<x<<endl;
Insert(root,x);
}
else
{
//cout<<Find(root,x)<<"aa"<<endl;
cout<<Find_k(root,x)<<endl;
}
//cout<<i<<":"<<endl;
//print(root);
//cout<<endl;
}
} void Rotate(Treap* &T,int f)
{
Treap* son=T->ch[f^1];//左旋处理的是右子树,而右旋处理的是左子树
T->ch[f^1]=son->ch[f];
son->ch[f]=T;
T->maintain();
son->maintain();
T=son;
} void Insert(Treap* &T,int value)
{
if (T==NULL)
T=new Treap(value);
else
{
int f=value<(T->key) ? 0 :1;
//cout<<value<<' '<<f<<' '<<(T->ch[0]==NULL)<<(T->ch[1]==NULL)<<endl;
Insert(T->ch[f],value);
if ((T->ch[f]->ran)>(T->ran))
Rotate(T,f^1);//如果是右子树则左旋,如果是左子树则右旋
}
T->maintain();
} int Find(Treap * T,int value)
{
while (T!=NULL)
{
//cout<<"Find_In"<<endl;
int f=T->compare(value);
if (f==-1)
return 1;
T=T->ch[f];
}
return 0;
} int Find_k(Treap * T,int value)
{
//cout<<"In"<<endl;
//cout<<T->ch[0]<<' '<<T->ch[1]<<endl;
//int Ans=T->key;
//cout<<"Init_Ans:"<<Ans<<endl;
int Ans=-inf;
while (T!=NULL)
{
//cout<<"Find_k :"<<T->key<<endl;
//cout<<value<<' '<<T->key<<endl;
if (T->key<=value)
Ans=max(Ans,T->key);
int f=T->compare(value);
if (f==-1) return value;
T=T->ch[f];
//Ans=T->key;
}
return Ans;
} void Delete(Treap* &T,int value)
{
int f=T->compare(value);
if (f==-1)
{
Treap* &T2=T;//因为后面要修改T指向,所以先用一个T2存下指针
if (T->ch[0]==NULL)
{
T=T->ch[1];
delete T2;
T2=NULL;
}
else
if (T->ch[1]==NULL)
{
T=T->ch[0];
delete T2;
T2=NULL;
}
else
{
int f2=T->ch[0]->ran > T->ch[1]->ran ? 1:0;
Rotate(T,f2);
Delete(T->ch[f2],value);
}
}
else Delete(T->ch[f],value);
if (T!=NULL)
T->maintain();
} void print(Treap *T)
{
if (T==NULL)
return;
cout<<T->key<<'(';
print(T->ch[0]);
cout<<',';
print(T->ch[1]);
cout<<')';
return;
}

Hihocoder 1325 平衡树·Treap(平衡树,Treap)的更多相关文章

  1. HihoCoder 1325 平衡树·Treap

    HihoCoder 1325 平衡树·Treap 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说 ...

  2. [BZOJ3223]文艺平衡树 无旋Treap

    3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB Description 您需要写一种数据结构(可参考题目标题),来维护一个 ...

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

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

  4. 【HIHOCODER 1325】 平衡树·Treap

    描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说的是哪两个啊? 小Ho:就是二叉排序树和堆啊,你看这两种数据结构都是构造了一个二叉树,一个节点有一个父亲和两个儿子. 如果 ...

  5. BZOJ3223文艺平衡树——非旋转treap

    此为平衡树系列第二道:文艺平衡树您需要写一种数据结构,来维护一个有序数列,其中需要提供以下操作: 翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 ...

  6. BZOJ3224普通平衡树——非旋转treap

    题目: 此为平衡树系列第一道:普通平衡树您需要写一种数据结构,来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3. 查询x数的排名(若有多个相同的数, ...

  7. BZOJ3786星系探索——非旋转treap(平衡树动态维护dfs序)

    题目描述 物理学家小C的研究正遇到某个瓶颈. 他正在研究的是一个星系,这个星系中有n个星球,其中有一个主星球(方便起见我们默认其为1号星球),其余的所有星球均有且仅有一个依赖星球.主星球没有依赖星球. ...

  8. BZOJ3159决战——树链剖分+非旋转treap(平衡树动态维护dfs序)

    题目描述 输入 第一行有三个整数N.M和R,分别表示树的节点数.指令和询问总数,以及X国的据点. 接下来N-1行,每行两个整数X和Y,表示Katharon国的一条道路. 接下来M行,每行描述一个指令或 ...

  9. BZOJ3729Gty的游戏——阶梯博弈+巴什博弈+非旋转treap(平衡树动态维护dfs序)

    题目描述 某一天gty在与他的妹子玩游戏.妹子提出一个游戏,给定一棵有根树,每个节点有一些石子,每次可以将不多于L的石子移动到父节点,询问将某个节点的子树中的石子移动到这个节点先手是否有必胜策略.gt ...

随机推荐

  1. linux新学篇

    [学会使用快捷键] Ctrl + C:这个是用来终止当前命令的快捷键,当然你也可以输入一大串字符,不想让它运行直接Ctrl + C,光标就会跳入下一行. Tab: 这个键是最有用的键了,也是笔者敲击概 ...

  2. 以太坊的crypto模块--以太坊源码学习

    以太坊的crypto模块 该模块分为两个部分一个是实现sha3,一个是实现secp256k1(这也是比特币中使用的签名算法). 需要说明的是secp256k1有两种实现方式,一种是依赖libsecp2 ...

  3. 关于 IDEA 自动识别问题,jsp页面Controller路径自动识别的问题

    idea之所以强大,就是强大的代码提示和联想功能,写起代码来简直不要太爽.但是这几天我发现在我的jsp页面中访问controller路径的时候不会自动提示了,对于这么严谨的我肯定要找出原因啊,哈哈. ...

  4. OpenGL ES 2.0 渲染管线 学习笔记

    图中展示整个OpenGL ES 2.0可编程管线 图中Vertex Shader和Fragment Shader 是可编程管线: Vertex Array/Buffer objects 顶点数据来源, ...

  5. String、StringBuffer、StringBuilder比较

    String.StringBuffer.StringBuilder三者是字符串中重要的内容,也是面试过程中经常问到的问题,下面就来总结一下三者的区别. 1.三者都可以存储和操作字符串. 2.Strin ...

  6. 【Selenium】Selenium1

    一.Selenium1组件 (1)Selenium服务器,负责启动关闭浏览器:解释和运行从测试程序中传来的Selenium命令:HTTP代理:获取和验证在浏览器和被测试的应用程序之间的传递的HTTP消 ...

  7. Nmap脚本引擎原理

    Nmap脚本引擎原理 一.NSE介绍 虽然Nmap内嵌的服务于版本探测已足够强大,但是在某些情况下我们需要多伦次的交互才能够探测到服务器的信息,这时候就需要自己编写NSE插件实现这个功能.NSE插件能 ...

  8. C++获取系统当前时间

    1.利用系统函数,不仅可以查看系统时间,而且还能修改系统时间 #include<stdlib.h> #include<iostream> using namespace std ...

  9. An overnight dance in discotheque

    An overnight dance in discotheque time limit per test 2 seconds memory limit per test 256 megabytes ...

  10. Redis客户端管理工具的安装及使用

    1.下载及安装 请到官网下载:www.treesoft.cn,要最新的版本treeNMS, window系统下载直接解压,就可以用了,免安装,免布署. 2.登录及连接参数配置 登录后,要配置连接参数信 ...