Treap
treap 是一种弱平衡的二叉搜索树。treap 这个单词是 tree 和 heap 的组合,表明 treap 是一种由树和堆组合形成的数据结构。treap 的每个结点上要额外储存一个值
treap 分为旋转式和无旋式两种。两种 treap 都易于编写,但无旋式 treap 的操作方式使得它天生支持维护序列、可持久化等特性。这里以重新实现 set<int>
(不可重集合)为例,介绍无旋式 treap。
无旋式 treap 的核心操作¶
无旋式 treap 又称分裂合并 treap。它仅有两种核心操作,即为分裂与合并。下面逐一介绍这两种操作。
分裂(split)¶
分裂过程接受两个参数:根指针
pair<node *, node *> split(node *u, int key) {
if (u == nullptr) {
return make_pair(nullptr, nullptr);
}
if (key < u->key) {
pair<node *, node *> o = split(u->lch, key);
u->lch = o.second;
return make_pair(o.first, u);
} else {
pair<node *, node *> o = split(u->rch, key);
u->rch = o.first;
return make_pair(u, o.second);
}
}
合并(merge)¶
合并过程接受两个参数:左 treap 的根指针
node *merge(node *u, node *v) {
if (u == nullptr) {
return v;
}
if (v == nullptr) {
return u;
}
if (u->priority > v->priority) {
u->rch = merge(u->rch, v);
return u;
} else {
v->lch = merge(u, v->lch);
return v;
}
}
建树(build)¶
将一个有
可以依次暴力插入这
在某些题目内,可能会有多次插入一段有序序列的操作,这是就需要在
方法一:在递归建树的过程中,每次选取当前区间的中点作为该区间的树根,并对每个节点钦定合适的优先值,使得新树满足堆的性质。这样能保证树高为
方法二:在递归建树的过程中,每次选取当前区间的中点作为该区间的树根,然后给每个节点一个随机优先级。这样能保证树高为 merge
操作更加随机一点,而不是用来保证树高的。
方法三:观察到 treap 是笛卡尔树,利用笛卡尔树的
将 treap 包装成为 set<int>
¶
count 函数¶
直接依靠二叉搜索树的性质查找即可。
int find(node *u, int key) {
if (u == nullptr) {
return 0;
}
if (key == u->key) {
return 1;
}
if (key < u->key) {
return find(u->lch, key);
} else {
return find(u->rch, key);
}
}
int count(int key) { return find(root, key); }
insert 函数¶
先在待插入的关键值处将整棵 treap 分裂,判断关键值是否已插入过之后新建一个结点,包含待插入的关键值,然后进行两次合并操作即可。
void insert(int key) {
pair<node*, node*> o = split(root, key);
if (find(root, key) == 0) {
o.first = merge(o.first, new node(key));
}
root = merge(o.first, o.second);
}
erase 函数¶
将具有待删除的关键值的结点从整棵 treap 中孤立出来(进行两侧分裂操作),删除中间的一段(具有待删除关键值),再将左右两端合并即可。
void erase(int key) {
pair<node*, node*> o = split(root, key - 1);
pair<node*, node*> p = split(o.second, key);
delete p.first;
root = merge(o.first, p.second);
}
旋转 treap¶
旋转 treap 在做普通平衡树题的时候,是所有平衡树中常数较小的
维护平衡的方式为旋转。性质与普通二叉搜索树类似
因为普通的二叉搜索树会被递增或递减的数据卡,用 treap 对每个节点定义一个权值,由 rand 得到,从而防止特殊数据卡。
每次删除/插入时通过 rand 值决定要不要旋转即可,其他操作与二叉搜索树类似
以下是 bzoj 普通平衡树模板代码
#include <algorithm>
#include <cstdio>
#include <iostream>
#define maxn 100005
#define INF (1 << 30)
int n;
struct treap {
int l[maxn], r[maxn], val[maxn], rnd[maxn], size[maxn], w[maxn];
int sz, ans, rt;
inline void pushup(int x) { size[x] = size[l[x]] + size[r[x]] + w[x]; }
void lrotate(int &k) {
int t = r[k];
r[k] = l[t];
l[t] = k;
size[t] = size[k];
pushup(k);
k = t;
}
void rrotate(int &k) {
int t = l[k];
l[k] = r[t];
r[t] = k;
size[t] = size[k];
pushup(k);
k = t;
}
void insert(int &k, int x) {
if (!k) {
sz++;
k = sz;
size[k] = 1;
w[k] = 1;
val[k] = x;
rnd[k] = rand();
return;
}
size[k]++;
if (val[k] == x) {
w[k]++;
} else if (val[k] < x) {
insert(r[k], x);
if (rnd[r[k]] < rnd[k]) lrotate(k);
} else {
insert(l[k], x);
if (rnd[l[k]] < rnd[k]) rrotate(k);
}
}
bool del(int &k, int x) {
if (!k) return false;
if (val[k] == x) {
if (w[k] > 1) {
w[k]--;
size[k]--;
return true;
}
if (l[k] == 0 || r[k] == 0) {
k = l[k] + r[k];
return true;
} else if (rnd[l[k]] < rnd[r[k]]) {
rrotate(k);
return del(k, x);
} else {
lrotate(k);
return del(k, x);
}
} else if (val[k] < x) {
bool succ = del(r[k], x);
if (succ) size[k]--;
return succ;
} else {
bool succ = del(l[k], x);
if (succ) size[k]--;
return succ;
}
}
int queryrank(int k, int x) {
if (!k) return 0;
if (val[k] == x)
return size[l[k]] + 1;
else if (x > val[k]) {
return size[l[k]] + w[k] + queryrank(r[k], x);
} else
return queryrank(l[k], x);
}
int querynum(int k, int x) {
if (!k) return 0;
if (x <= size[l[k]])
return querynum(l[k], x);
else if (x > size[l[k]] + w[k])
return querynum(r[k], x - size[l[k]] - w[k]);
else
return val[k];
}
void querypre(int k, int x) {
if (!k) return;
if (val[k] < x)
ans = k, querypre(r[k], x);
else
querypre(l[k], x);
}
void querysub(int k, int x) {
if (!k) return;
if (val[k] > x)
ans = k, querysub(l[k], x);
else
querysub(r[k], x);
}
} T;
int main() {
srand(123);
scanf("%d", &n);
int opt, x;
for (int i = 1; i <= n; i++) {
scanf("%d%d", &opt, &x);
if (opt == 1)
T.insert(T.rt, x);
else if (opt == 2)
T.del(T.rt, x);
else if (opt == 3) {
printf("%d\n", T.queryrank(T.rt, x));
} else if (opt == 4) {
printf("%d\n", T.querynum(T.rt, x));
} else if (opt == 5) {
T.ans = 0;
T.querypre(T.rt, x);
printf("%d\n", T.val[T.ans]);
} else if (opt == 6) {
T.ans = 0;
T.querysub(T.rt, x);
printf("%d\n", T.val[T.ans]);
}
}
return 0;
}
练习题¶
buildLast update and/or translate time of this article,Check the history
editFound smelly bugs? Translation outdated? Wanna contribute with us? Edit this Page on Github
peopleContributor of this article Dev-XYS
translateTranslator of this article Visit the original article!
copyrightThe article is available under CC BY-SA 4.0 & SATA ; additional terms may apply.