8.1 伸展树

局部性(Locality):刚被访问过的数据,极有可能很快再次被访问
逐层伸展:节点v一旦被访问,随即转移至根
自上而上,逐层单旋(zig, v->parent; zag, v->parent),直至v最终被推送至根
最坏情况:旋转次数呈周期性算术级数演变,每一周期累计Ω(n2),分摊Ω(n)
双层伸展:向上追溯两层,反复考察三代g = parent(p), p = parent(v), v,根据其相对位置,经两次旋转使得v上升两层,成为子树根
zig-zag / zag-zig,与AVL树双旋完全等效,与逐层伸展一致
zig-zig / zag-zag,一旦访问坏节点,对应路径长度将随即减半(折叠效果),最坏情况不致持续发生,单次伸展操作分摊O(logn)
zig / zag,此时必有parent(v) == root(T),且每轮调整中至多(在最后)出现一次该情况

接口

template <typename T> class Splay: public BST<T> { // 由BST派生
protected:
    BinNodePosi(T) splay(BinNodePosi(T) v); // 将v伸展至根
public:
    BinNodePosi(T) & search(const T & e); // 查找重写
    BinNodePosi(T) insert(const T & e); // 插入重写
    bool remove(const T & e); // 删除重写
}

伸展算法

template <typename T> BinNodePosi(T)::splay(BinNodePosi(T) v) {
    if (!v)
        return NULL;
    BinNodePosi(T) p;
    BinNodePosi(T) g;
    while ((p = v->parent) && (g = p->parent)) { // 自下而上,反复双层伸展
        BinNodePosi(T) gg = g->parent;
        if (IsLChild(*v)) {
            if (IsLChild(*p)) { // zig-zig
                attachAsLChild(g, p->rc);
                attachAsLChild(p, v->rc);
                attachAsRChild(p, g);
                attachAsRChild(v, p);
            } else { // zig-zag
                attachAsLChild(p, v->rc);
                attachAsRChild(g, v->lc);
                attachAsLChild(v, g);
                attachAsRChild(v, p);
            }
        } else {
            if (IsRChild(*p)) { // zag-zag
                attachAsRChild(g, p->lc);
                attachAsRChild(p, v->lc);
                attachAsLChild(p, g);
                attachAsLChild(v, p);
            } else { // zag-zig
                attachAsRChild(p, v->lc);
                attachAsLChild(g, v->rc);
                attachAsRChild(v, g);
                attachAsLChild(v, p)
            }
        }
        if (!gg)
            v->parent = NULL;
        else
            (g == gg->lc) ? attachAsLChild(gg, v) : attachAsRChild(gg, v);
        updateHeight(g);
        updateHeight(p);
        updateHeight(v);
    } // 双层伸展结束时必有g == NULL,但p可能非空
    if (p = v->parent) {
        /* 若p为根,则只需再单旋至多一次 */
    }
    v->parent = NULL;
    return v;
}

查找算法

template <typename T> BinNodePosi(T) & Splay<T>::search(const T & e) {
    BinNodePosi(T) p = searchIn(_root, e, _hot=NULL); // 调用标准BST内部接口定位目标节点
    _root = splay(p ? p : _hot); // 无论成功与否,最后被访问的节点将伸展至根
    return _root; // 总是返回根节点
} // 伸展树查找操作与常规BST::search()不同,可能改变树的拓扑结构,不属于静态操作

插入算法

template <typename T> BinNodePosi(T) Splay<T>::insert(const T & e) {
    if (!_root) { // 处理原树为空的退化情况
        _size++;
        return _root = new BinNode<T>(e);
    }
    if (e == search(e)->data) // 确认目标节点不存在
        return _root;
    _size++;
    BinNodePosi(T) t = _root; // 创建新节点
    if (_root->data < e) { // 插入新根节点,以t与t->rChild为左右子节点
        t->parent = _root = new BinNode<T>(e, NULL, t, t->rChild);
        if (HasRChild(*t)) {
            t->rChild->parent = _root;
            t->rChild = NULL;
        }
    } else { // 插入新根节点,以t->lChild与t为左右子节点
        t->parent = _root = new BinNode<T>(e, NULL, t->lChild, t);
        if (HasLChild(*t)) {
            t->lChild->parent = _root;
            t->lChild = NULL;
        }
    }
    updateHeightAbove(t); // 更新高度
    return _root; // 新节点必然位于树根,返回
}

删除算法

template <typename T> bool Splay<T>::remove(const T& e) {
    if (!_root || (e != search(e)->data))
        return false; // 若树为空或目标不存在,则无法删除
    BinNodePosi(T) w = _root; // 经search()后节点e已被伸展至树根
    if (!HasLChild(*_root)) { // 若无左子树,则直接删除
        _root = _root->rChild;
        if (_root)
            _root->parent = NULL;
    } else if (!HasRChild(*_root)) { // 若无右子树,则直接删除
        _root = _root->lChild;
        if (_root)
            _root->parent = NULL;
    } else { // 若左右子树同时存在
        BinNodePosi(T) lTree = _root->lChild;
        lTree->parent = NULL;
        _root->lChild = NULL; // �暂时删除左子树
        _root = _root->rChild;
        _root->parent = NULL; // 保留右子树
        search(w->data); // 以原树根为目标进行查找(必失败),至此右子树中最小节点必伸展至根,且因无相同节点,左子树必空
        _root->lChild = lTree;
        lTree->parent = _root; // 左子树接回原位
    }
    release(w->data);
    release(w);
    _size--; // 释放节点,更新规模
    if (_root)
        updateHeight(_root); // 若树非空,则更新树根高度
    return true;
}

伸展树无需记录节点高度或平衡因子,编程实现简单易行,优于AVL树
分摊复杂度O(logn),与AVL树相当
局部性强,缓存命中率极高时(k << n << m),效率可以更高,任何连续的m次查找,可在O(mlogk + nlogn)时间内完成
仍无法保证单次最坏情况出现,不适用于对效率敏感的场合

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342