链接一下题目:luoguP3369[模板]普通平衡树(Treap/SBT)

平衡树解析

#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<iomanip>
#include<algorithm>
#include<ctime>
#include<queue>
#include<stack>
#define lst long long
#define rg register
#define N 500050
#define Inf 2147483647
using namespace std; int use,root,tot;//操作数,根节点编号,树中元素总和
struct T{
int cnt;//这个数相等的数的数量
int size;//这棵子树上一共有几个元素
int fa,v;//父亲节点,当前点的权值
int ch[];//左(0)右(1)孩子
}ljl[N];//平衡树 inline int read()
{
rg int s=,m=;char ch=getchar();
while(ch!='-'&&(ch<''||ch>''))ch=getchar();
if(ch=='-')m=-,ch=getchar();
while(ch>=''&&ch<='')s=(s<<)+(s<<)+ch-'',ch=getchar();
return s*m;
} inline void Pushup(rg int now)//更新节点size的操作
{
ljl[now].size=ljl[ljl[now].ch[]].size+ljl[ljl[now].ch[]].size+ljl[now].cnt;
//当前节点的size是左孩子子树的size加上右孩子子树的size加上本节点相等的数的数量
} inline void rotate(rg int x)//把x往上转
{//定义:x的相对位置为x属于y的?孩子
rg int y=ljl[x].fa;//父亲
rg int z=ljl[y].fa;//祖父
rg int k=ljl[y].ch[]==x;//x的相对位置
ljl[z].ch[ljl[z].ch[]==y]=x;//把x转到y的位置上去
ljl[x].fa=z;//x的爸爸变成了z
ljl[y].ch[k]=ljl[x].ch[k^];//y的x的相对位置的那个孩子变成x的x的相对位置的另一个孩子
ljl[ljl[x].ch[k^]].fa=y;//……的爸爸变成y
ljl[x].ch[k^]=y;//x的相对位置的另一个孩子变成y
ljl[y].fa=x;//y的爸爸变成x
Pushup(x),Pushup(y);//更新一下节点数量
} inline void splay(rg int x,rg int goal)//把x转到goal下面,如果goal=0,那么就是转到根节点
{
while(ljl[x].fa!=goal)//如果x的父亲不是goal,目标没有达成,就要继续转
{
rg int y=ljl[x].fa;//父亲
rg int z=ljl[y].fa;//祖父
if(z!=goal)//如果z存在的话
{
(x==ljl[y].ch[])^(y==ljl[z].ch[])?rotate(x):rotate(y);
//如果x和y分别是y和z的同一孩子,就把y往上转
//如果x和y分别是y和z的不同孩子,就把x往上转
}
rotate(x);//最后一定会要把x在网上转一次
}
if(!goal)root=x;//更新根节点
} void Insert(rg int x)//插入x
{
rg int now=root,fa=;//从根开始找,根的父亲是0
while(ljl[now].v!=x&&now)//只要还没有找到这个数字,且当前这个位置有数,就继续找
{
fa=now;//爸爸变成现在的节点
now=ljl[now].ch[x>ljl[now].v];//如果x比now大,就找now的右孩子,小则左孩子
}
if(now)ljl[now].cnt++;//如果存值的位置存在,就直接在计数器上加1
else//否则
{
now=++tot;//增加一个新位置
if(fa)ljl[fa].ch[x>ljl[fa].v]=now;//如果父亲存在(我不是根),那我的父亲的儿子是我
ljl[now].v=x;//权值
ljl[now].fa=fa;//父亲
ljl[now].cnt=;//计数器
ljl[now].size=;//子树大小
ljl[now].ch[]=ljl[now].ch[]=;//没有孩子
}
splay(now,);//把当前位置转到根节点,以维持树的平衡
} inline void find(rg int x)//找x的位置,把它转到根节点,方便之后的计算
{
rg int now=root;//从根开始找
if(!root)return;//如果是空树,还找个屁
while(x!=ljl[now].v&&ljl[now].ch[x>ljl[now].v])//如果还没有找到那个点,且我还有符合的儿子
now=ljl[now].ch[x>ljl[now].v];//就跳转到我的儿子继续找
splay(now,);//转到树根去
} inline int Next(rg int x,rg int f)//找x的前驱(0)后继(1)
{
find(x);//先找到x的位置,可能树顶不是x,是和x值接近的那个元素
int now=root;//从根开始找
if(ljl[now].v>x&&f)return now;//如果大于x且我们要找后继,那就是他了
if(ljl[now].v<x&&!f)return now;//如果小于x且我们要找前驱,那就是他了
now=ljl[now].ch[f];//那我们从符合条件的儿子开始跳
while(ljl[now].ch[f^])now=ljl[now].ch[f^];//不断的往最优的方向跳
return now;//返回位置
} inline void Delete(rg int x)//删掉x
{
rg int qq=Next(x,);//找到前驱
rg int hj=Next(x,);//找到后继
splay(qq,),splay(hj,qq);//把前驱转到树根,把后继转到前驱下面
int del=ljl[hj].ch[];//那么x就是后继的左儿子
if(ljl[del].cnt>)//如果x的计数器大于1
{
ljl[del].cnt--;//让x的计数器--
splay(del,);//转到树根保持平衡
}
else ljl[hj].ch[]=;//直接删除x
} inline int kth(rg int x)//找第x小的数
{
rg int now=root;//从根开始找
if(ljl[now].size<x)return ;//如果排名都超过总数了…………
while()//嘿嘿,一直找
{
rg int ls=ljl[now].ch[];//左孩子
if(ljl[ls].size+ljl[now].cnt<x)//如果排名比左孩子总元素数和与我相等的数总和还大
{
x-=ljl[ls].size+ljl[now].cnt;//就减去前面的元素数
now=ljl[now].ch[];//去右孩子上找这个排名
}
else
if(ljl[ls].size>=x)now=ls;//如果左孩子里包括它
else return ljl[now].v;//那就在这个点上了,返回
}
} int main()
{
use=read();
Insert(Inf),Insert(-Inf);
for(rg int i=;i<=use;++i)
{
rg int opt=read(),x=read();
if(opt==)Insert(x);
if(opt==)Delete(x);
if(opt==)find(x),printf("%d\n",ljl[ljl[root].ch[]].size);
if(opt==)printf("%d\n",kth(x+));
if(opt==)printf("%d\n",ljl[Next(x,)].v);
if(opt==)printf("%d\n",ljl[Next(x,)].v);
}
return ;
}

luoguP3369[模板]普通平衡树(Treap/SBT) 题解的更多相关文章

  1. 【模板】平衡树——Treap和Splay

    二叉搜索树($BST$):一棵带权二叉树,满足左子树的权值均小于根节点的权值,右子树的权值均大于根节点的权值.且左右子树也分别是二叉搜索树.(如下) $BST$的作用:维护一个有序数列,支持插入$x$ ...

  2. [luogu P3369]【模板】普通平衡树(Treap/SBT)

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

  3. 洛谷P3369 【模板】普通平衡树(Treap/SBT)

    洛谷P3369 [模板]普通平衡树(Treap/SBT) 平衡树,一种其妙的数据结构 题目传送门 题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作: 插入x数 删除 ...

  4. AC日记——【模板】普通平衡树(Treap/SBT) 洛谷 P3369

    [模板]普通平衡树(Treap/SBT) 思路: 劳资敲了一个多星期: 劳资终于a了: 劳资一直不a是因为一个小错误: 劳资最后看的模板: 劳资现在很愤怒: 劳资不想谈思路!!! 来,上代码: #in ...

  5. 数组splay ------ luogu P3369 【模板】普通平衡树(Treap/SBT)

    二次联通门 : luogu P3369 [模板]普通平衡树(Treap/SBT) #include <cstdio> #define Max 100005 #define Inline _ ...

  6. 替罪羊树 ------ luogu P3369 【模板】普通平衡树(Treap/SBT)

    二次联通门 : luogu P3369 [模板]普通平衡树(Treap/SBT) 闲的没事,把各种平衡树都写写 比较比较... 下面是替罪羊树 #include <cstdio> #inc ...

  7. 红黑树 ------ luogu P3369 【模板】普通平衡树(Treap/SBT)

    二次联通门 : luogu P3369 [模板]普通平衡树(Treap/SBT) 近几天闲来无事...就把各种平衡树都写了一下... 下面是红黑树(Red Black Tree) 喜闻乐见拿到了luo ...

  8. P3369 【模板】普通平衡树 Treap

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

  9. 算法模板——平衡树Treap 2

    实现功能:同平衡树Treap 1(BZOJ3224 / tyvj1728) 这次的模板有了不少的改进,显然更加美观了,几乎每个部分都有了不少简化,尤其是删除部分,这个参照了hzwer神犇的写法,在此鸣 ...

随机推荐

  1. MongoDB的使用学习之(二)简介

    原文链接:http://www.cnblogs.com/yxlblogs/p/3681089.html MongoDB 是一个高性能,开源,无模式的文档型数据库,是当前 NoSQL 数据库产品中最热门 ...

  2. 基于Xilinx Kintex-7 FPGA K7 XC7K325T PCIeX8 四路光纤卡

    基于Xilinx Kintex-7 FPGA K7 XC7K325T PCIeX8 四路光纤卡 1. 板卡概述   板卡主芯片采用Xilinx公司的XC7K325T-2FFG900 FPGA,pin_ ...

  3. voc数据集坐标,coco数据集坐标

    voc,如上图 x1 ,y1 ,x4, y4    bbox的坐标格式是,x,y的最大最小值,也就是box的左上角和右下角的坐标 coco x,y,w,h       box左上角的坐标以及宽.高 图 ...

  4. POJ 3784 Running Median (动态中位数)

    题目链接:http://poj.org/problem?id=3784 题目大意:依次输入n个数,每当输入奇数个数的时候,求出当前序列的中位数(排好序的中位数). 此题可用各种方法求解. 排序二叉树方 ...

  5. w = tf.Variable(<initial-value>, name=<optional-name>)

    w = tf.Variable(<initial-value>, name=<optional-name>)

  6. 【leetcode】307. Range Sum Query - Mutable

    题目如下: 解题思路:就三个字-线段树.这个题目是线段树用法最经典的场景. 代码如下: class NumArray(object): def __init__(self, nums): " ...

  7. shell第一个脚本

    mkdir 创建目录touch 创建空文件 chmod +x ./test.sh  #使脚本具有执行权限

  8. [luogu]P1016 旅行家的预算[贪心]

    [luogu]P1016 旅行家的预算 题目描述 一个旅行家想驾驶汽车以最少的费用从一个城市到另一个城市(假设出发时油箱是空的).给定两个城市之间的距离D1.汽车油箱的容量C(以升为单位).每升汽油能 ...

  9. 『转』一千行MySQL学习笔记

    /* 启动MySQL */ net start mysql /* 连接与断开服务器 */ mysql -h 地址 -P 端口 -u 用户名 -p 密码 /* 跳过权限验证登录MySQL */ mysq ...

  10. 阿里云不支持stmp 的25端口,必须

    第一种方法 到阿里云解封25端口 特别注意阿里云的<25端口使用服务协议>:    我/我公司承诺并保证TCP 25端口仅用来连接第三方的SMTP服务器,从第三方的SMTP服务器外发邮件. ...