数据结构之伸展树详解(2)
(1)旋转操作
// node 为结点类型,其中ch[0]表示左结点指针,ch[1]表示右结点指针 // pre 表示指向父亲的指针 // Rotate函数用于(左/右)旋转x->pre void Rotate(node *x, int d) // 旋转操作,d=0 表示左旋,d=1 表示右旋 { node *y = x->pre; Push_Down(y), Push_Down(x); // 先将Y 结点的标记向下传递(因为Y 在上面),再把X 的标记向下传递 y->ch[! d] = x->ch[d]; if (x->ch[d] != Null) x->ch[d]->pre = y; x->pre = y->pre; if (y->pre != Null) if (y->pre->ch[0] == y) y->pre->ch[0] = x; else y->pre->ch[1] = x; x->ch[r] = y, y->pre = x, Update(y); // 维护Y 结点 if (y == root) root = x; // root 表示整棵树的根结点 }
(2)splay操作
void Splay(node *x, node *f) // Splay 操作,表示把结点x 转到结点f 的下面 { for (Push_Down(x) ; x->pre != f; ) // 一开始就将X 的标记下传 if (x->pre->pre == f) // 父结点的父亲即为f,执行单旋转 if (x->pre->ch[0] == x) Rotate(x, 1); else Rotate(x, 0); else { node *y = x->pre, *z = y->pre; if (z->ch[0] == y) if (y->ch[0] == x) Rotate(y, 1), Rotate(x, 1); // 一字形旋转 else Rotate(x, 0), Rotate(x, 1); // 之字形旋转 else if (y->ch[1] == x) Rotate(y, 0), Rotate(x, 0); // 一字形旋转 else Rotate(x, 1), Rotate(x, 0); // 之字形旋转 } Update(x); // 最后再维护X 结点 }
(3)将第k个数转到要求的位置
// 找到处在中序遍历第k 个结点,并将其旋转到结点f 的下面 void Select(int k, node *f) { int tmp; node *t; for (t = root; ; ) // 从根结点开始 { Push_Down(t); // 由于要访问t 的子结点,将标记下传 tmp = t->ch[0]->size; // 得到t 左子树的大小 if (k == tmp + 1) break; // 得出t 即为查找结点,退出循环 if (k <= tmp) // 第k 个结点在t 左边,向左走 t = t->ch[0]; else // 否则在右边,而且在右子树中,这个结点不再是第k 个 k -= tmp + 1, t = t->ch[1]; } Splay(t, f); // 执行旋转 }
5、 应用
(1)数列维护问题
题目:维护一个数列,支持以下几种操作:
1. 插入:在当前数列第posi 个数字后面插入tot 个数字;若在数列首位插入,则posi 为0。
2. 删除:从当前数列第posi 个数字开始连续删除tot 个数字。
3. 修改:从当前数列第posi 个数字开始连续tot 个数字统一修改为c 。
4. 翻转:取出从当前数列第posi 个数字开始的tot 个数字,翻转后放入原来的位置。
5. 求和:计算从当前数列第posi 个数字开始连续tot 个数字的和并输出。
6. 求和最大子序列:求出当前数列中和最大的一段子序列,并输出最大和。
(2)轻量级web服务器lighttpd中用到数据结构splay tree.
6、 参考资料
(1)杨思雨《伸展树的基本操作与应用》
(2)Crash《运用伸展树解决数列维护问题》
- 上一篇:数据结构之Treap详解
- 下一篇:数据结构之堆详解