Hihocoder 1325 平衡树·Treap(平衡树,Treap)
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)的更多相关文章
- HihoCoder 1325 平衡树·Treap
HihoCoder 1325 平衡树·Treap 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说 ...
- [BZOJ3223]文艺平衡树 无旋Treap
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MB Description 您需要写一种数据结构(可参考题目标题),来维护一个 ...
- bzoj3224: Tyvj 1728 普通平衡树(平衡树)
bzoj3224: Tyvj 1728 普通平衡树(平衡树) 总结 a. cout<<(x=3)<<endl;这句话输出的值是3,那么对应的,在splay操作中,当父亲不为0的 ...
- 【HIHOCODER 1325】 平衡树·Treap
描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说的是哪两个啊? 小Ho:就是二叉排序树和堆啊,你看这两种数据结构都是构造了一个二叉树,一个节点有一个父亲和两个儿子. 如果 ...
- BZOJ3223文艺平衡树——非旋转treap
此为平衡树系列第二道:文艺平衡树您需要写一种数据结构,来维护一个有序数列,其中需要提供以下操作: 翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 ...
- BZOJ3224普通平衡树——非旋转treap
题目: 此为平衡树系列第一道:普通平衡树您需要写一种数据结构,来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3. 查询x数的排名(若有多个相同的数, ...
- BZOJ3786星系探索——非旋转treap(平衡树动态维护dfs序)
题目描述 物理学家小C的研究正遇到某个瓶颈. 他正在研究的是一个星系,这个星系中有n个星球,其中有一个主星球(方便起见我们默认其为1号星球),其余的所有星球均有且仅有一个依赖星球.主星球没有依赖星球. ...
- BZOJ3159决战——树链剖分+非旋转treap(平衡树动态维护dfs序)
题目描述 输入 第一行有三个整数N.M和R,分别表示树的节点数.指令和询问总数,以及X国的据点. 接下来N-1行,每行两个整数X和Y,表示Katharon国的一条道路. 接下来M行,每行描述一个指令或 ...
- BZOJ3729Gty的游戏——阶梯博弈+巴什博弈+非旋转treap(平衡树动态维护dfs序)
题目描述 某一天gty在与他的妹子玩游戏.妹子提出一个游戏,给定一棵有根树,每个节点有一些石子,每次可以将不多于L的石子移动到父节点,询问将某个节点的子树中的石子移动到这个节点先手是否有必胜策略.gt ...
随机推荐
- PHP版本替换, phpinfo和php -v显示版本信息不一致
环境:OS X EI Capitan 10.11 & lnmp 背景: 1想将lamp(xampp安装的,php5.2)换成 lnmp(php7.0) 2php5.2卸载(xampp卸载& ...
- Python 操作 Azure Blob Storage
笔者在<Azure 基础:Blob Storage>一文中介绍了 Azure Blob Storage 的基本概念,并通过 C# 代码展示了如何进行基本的操作.最近笔者需要在 Linux ...
- (转)关于BigDecimal 转化字符串toPlainString()和toString()的区别
对于 BigDecimal b ; (b=(0.4321)^ 20)String s = b.toPlainString() ;System.out.println(s) ; 输出为:0.0 ...
- python编写知乎爬虫实践
爬虫的基本流程 网络爬虫的基本工作流程如下: 首先选取一部分精心挑选的种子URL 将种子URL加入任务队列 从待抓取URL队列中取出待抓取的URL,解析DNS,并且得到主机的ip,并将URL对应的网页 ...
- oracle 11g 完全卸载方法
网上好多卸载教程都前篇一律,但很多卸完重装都有问题,卸了几次装了几次,就特地总结整理一下 另外说一句:在完全删除(或者叫卸载)oracle时,没有必要特别意oracle提示问题,只要把oracle痕迹 ...
- 1.JAVA WEB 笔记中文乱码
JAVA WEB 乱码问题解析 乱码原因 在Java Web开发过程中,经常遇到乱码的问题,造成乱码的原因,概括起来就是对字符编码和解码的方式不匹配. 既然乱码的原因是字符编码与解码的方式不匹配,那么 ...
- thinkphp5.0学习笔记(四)数据库的操作
ThinkPHP内置了抽象数据库访问层,把不同的数据库操作封装起来,我们只需要使用公共的Db类进行操作,而无需针对不同的数据库写不同的代码和底层实现,Db类会自动调用相应的数据库驱动来处理.采用PDO ...
- [leetcode-617-Merge Two Binary Trees]
Given two binary trees and imagine that when you put one of them to cover the other, some nodes of t ...
- cvCvtColor与cvtColor区别
用到了rgb转灰度图功能,查到两个函数,发现名字很像,功能也一样,但是参数类型不一样. 记录一下. 可以看声明,cvCvtColor是c语言风格接口. /* Converts input array ...
- 【LeetCode】284. Peeking Iterator
题目: Given an Iterator class interface with methods: next() and hasNext(), design and implement a Pee ...