Size Balance Tree(SBT模板整理)
/*
* tree[x].left 表示以 x 为节点的左儿子
* tree[x].right 表示以 x 为节点的右儿子
* tree[x].size 表示以 x 为根的节点的个数(大小)
*/
struct SBT
{
int key,left,right,size;
} tree[10010];
int root = 0,top = 0;
void left_rot(int &x) // 左旋
{
int y = tree[x].right;
if (!y) return;
tree[x].right = tree[y].left;
tree[y].left = x;
tree[y].size = tree[x].size;
tree[x].size = tree[tree[x].left].size + tree[tree[x].right].size + 1;
x = y;
}
void right_rot(int &x) //右旋
{
int y = tree[x].left;
if (!y) return;
tree[x].left = tree[y].right;
tree[y].right = x;
tree[y].size = tree[x].size;
tree[x].size = tree[tree[x].left].size + tree[tree[x].right].size + 1;
x = y;
}
void maintain(int &x,bool flag) //维护SBT状态
{
if (!x) return;
if (flag == false) //左边
{
if(tree[tree[tree[x].left].left].size > tree[tree[x].right].size)//左孩子的左孩子大于右孩子
right_rot(x);
else if (tree[tree[tree[x].left].right].size > tree[tree[x].right].size) //左孩子的右孩子大于右孩子
{
left_rot(tree[x].left);
right_rot(x);
}
else
return;
}
else //右边
{
if(tree[tree[tree[x].right].right].size > tree[tree[x].left].size)//右孩子的右孩子大于左孩子
left_rot(x);
else if(tree[tree[tree[x].right].left].size > tree[tree[x].left].size) //右孩子的左孩子大于左孩子
{
right_rot(tree[x].right);
left_rot(x);
}
else
return;
}
maintain(tree[x].left,false);
maintain(tree[x].right,true);
maintain(x,true);
maintain(x,false);
}
void insert(int &x,int key) //插入
{
if (x == 0)
{
x = ++top;
tree[x].left = 0;
tree[x].right = 0;
tree[x].size = 1;
tree[x].key = key;
}
else
{
tree[x].size++;
if(key < tree[x].key)
insert(tree[x].left,key);
else
insert(tree[x].right,key);//相同元素可插右子树
maintain(x,key >= tree[x].key);
}
}
int remove(int &x,int key) //利用后继删除
{
tree[x].size--;
if(key > tree[x].key)
remove(tree[x].right,key);
else if(key < tree[x].key)
remove(tree[x].left,key);
else if(tree[x].left !=0 && !tree[x].right) //有左子树,无右子树
{
int tmp = x;
x = tree[x].left;
return tmp;
}
else if(!tree[x].left && tree[x].right != 0) //有右子树,无左子树
{
int tmp = x;
x = tree[x].right;
return tmp;
}
else if(!tree[x].left && !tree[x].right) //无左右子树
{
int tmp = x;
x = 0;
return tmp;
}
else //左右子树都有
{
int tmp = tree[x].right;
while(tree[tmp].left)
tmp = tree[tmp].left;
tree[x].key = tree[temp].key;
remove(tree[x].right,tree[tmp].key);
}
}
int getmin(int x) //求最小值
{
while(tree[x].left)
x = tree[x].left;
return tree[x].key;
}
int getmax(int x) //求最大值
{
while(tree[x].right)
x = tree[x].right;
return tree[x].key;
}
int pred(int &x,int y,int key) //前驱,y初始前驱,从0开始, 最终要的是返回值的key值
{
if(x == 0)
return y;
if(key > tree[x].key)
return pred(tree[x].right,x,key);
else
return pred(tree[x].left,y,key);
}
int succ(int &x,int y,int key) //后继,同上
{
if(x == 0)
return y;
if(key < tree[x].key)
return succ(tree[x].left,x,key);
else
return succ(tree[x].right,y,key);
}
int select(int &x,int k) //查找第k小的数
{
int r = tree[tree[x].left].size + 1;
if(r == k)
return tree[x].key;
else if(r < k)
return select(tree[x].right,k - r);
else
return select(tree[x].left,k);
}
int rank(int &x,int key) //key排第几
{
if(key < tree[x].key)
{
return rank(tree[x].left,key);
}
else if(key > tree[x].key)
return rank(tree[x].right,key) + tree[tree[x].left].size + 1;
else
return tree[tree[x].left].size + 1;
}
int main()
{
//insert(root,key);
//delete(root,key)
return 0;
}
详情查看陈启峰大神的论文
Size Balance Tree(SBT模板整理)的更多相关文章
- Size Balanced Tree(SBT树)整理
不想用treap和Splay,那就用SB树把,哈哈,其实它一点也SB,厉害着呢. 先膜拜一下作者陈启峰.Orz 以下内容由我搜集整理得来. 一.BST及其局限性 二叉查找树(Binary Search ...
- 初学 Size Balanced Tree(bzoj3224 tyvj1728 普通平衡树)
SBT(Size Balance Tree), 即一种通过子树大小(size)保持平衡的BST SBT的基本性质是:每个节点的size大小必须大于等于其兄弟的儿子的size大小: 当我们插入或者删除一 ...
- Size Balanced Tree
Size Balanced Tree(SBT)是目前速度最快的平衡二叉搜索树,且能够进行多种搜索操作,区间操作:和AVL.红黑树.伸展树.Treap类似,SBT也是通过对节点的旋转来维持树的平衡,而相 ...
- Size Balanced Tree(SBT) 模板
首先是从二叉搜索树开始,一棵二叉搜索树的定义是: 1.这是一棵二叉树: 2.令x为二叉树中某个结点上表示的值,那么其左子树上所有结点的值都要不大于x,其右子树上所有结点的值都要不小于x. 由二叉搜索树 ...
- ACM算法模板整理
史诗级ACM模板整理 基本语法 字符串函数 istream& getline (char* s, streamsize n ); istream& getline (char* s, ...
- C基础 - 终结 Size Balanced Tree
引言 - 初识 Size Balanced Tree 最近在抽细碎的时间看和学习 random 的 randnet 小型网络库. iamrandom/randnet - https://github. ...
- 字符串系列——KMP模板整理
KMP模板整理 KMP与扩展KMP: /*vs 2017/ vs code以外编译器,去掉windows.h头文件和system("pause");*/ #include<i ...
- 子树大小平衡树(Size Balanced Tree,SBT)操作模板及杂谈
基础知识(包括但不限于:二叉查找树是啥,SBT又是啥反正又不能吃,平衡树怎么旋转,等等)在这里就不(lan)予(de)赘(duo)述(xie)了. 先贴代码(数组模拟): int seed; int ...
- Link-Cut Tree指针模板
模板: 以下为弹飞绵羊代码: #define Troy #include "bits/stdc++.h" using namespace std; ; inline int rea ...
随机推荐
- sun.misc.Unsafe的理解
以下sun.misc.Unsafe源码和demo基于jdk1.7: 最近在看J.U.C里的源码,很多都用到了sun.misc.Unsafe这个类,一知半解,看起来总感觉有点不尽兴,所以打算对Unsaf ...
- ADO.NET五大对象理论和实践(草稿)
一.ADO.NET五大对象理论 1. Connection:与数据源建立连接. 2. Command:对数据源执行SQL命令并返回结果. Command对象在执行的的时候有几个比较重要的方法,如Exe ...
- 使用TortoiseGit 来进行日常操作
TortoiseSVN大家应该都很熟悉,相应的,git也有一个工具是TortoiseGit. 下载地址: http://download.tortoisegit.org/tgit/previews/ ...
- js输出二维数组最长的子数组
,,],[,,,],[,,,,]]; ].length; ; i < a.length; i++) { if (max<a[i].length) { max=a[i].length; va ...
- 解惑spring嵌套事物
工作中一直对spring中的事物管理都是最简单的配置 但是spring中的事物传播性配置 还有很多种,有时候经常疑惑service调用service的问题,今天的论坛上看到一篇写的非常详细的文字.记录 ...
- jsp/servlet 中sendRedirect,include,forward区别
1 sendRedirect response.sendRedirect(); 服务器根据逻辑,发送一个状态码,告诉浏览器重新去请求新的地址,一般来说浏览器会用刚才请求的所有参数重新请求,所以sess ...
- amCharts图表中的JavaScript中文注释引起的浏览器兼容性问题
近期用amCharts做图表.一切都很顺利,然后演示的时候掉链子了,平时开发的时候都是用的火狐和谷歌,加上这种图表框架本来就号称兼容性极好,也没有在ie上测试,演示的机器上恰巧用的是ie11,发现一个 ...
- NYOJ 743
复杂度 描述 for(i=1;i<=n;i++) for(j=i+1;j<=n;j++) for(k=j+1;k<=n;k++) operation; 你知道 operation 共 ...
- Hibernate考试选择题解析
1.在Hibernate中,以下关于主键生成器说法错误的是(AC). A.increment可以用于类型为long.short或byte的主键(byte类型不可以) B.identity用于如SQL ...
- Socket
Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求. 以J2SDK-1.3为例,Socket和ServerSocket类库位于 ...