7.2 BBST

理想平衡
节点数目固定时,兄弟子树高度越接近(平衡),全树越趋向于更低
由n个节点组成二叉树,高度不低于\left \lfloor \log_{2}{n} \right \rfloor,恰为\left \lfloor \log_{2}{n} \right \rfloor时,称作理想平衡
理想平衡出现概率极低,维护成本过高
高度渐进不超过O(logn),即可称作适度平衡
适度平衡的BST,称作平衡二叉搜索树(BBST)

等价BST
联接关系不尽相同,承袭关系可能颠倒
中序遍历序列完全一致,全局单调非降

平衡因子
balFac(v) = height(lc(v)) - height(rc(v))
G.Adelson-Velsky & E. Landis: ∀ v, | balcFac(v) | ≤ 1
height(AVL) = O(logn)
高度为h的AVL树,至少包含S(h) = fib(h + 3) - 1个节点

AVL接口

#define Balanced(x) \ // 理想平衡
    (stature((x).lChild) == stature((x).rChild))
#define BalFac(x) \ // 平衡因子
    (stature((x).lChild) - stature((x).rChild))
#define AvlBalanced(x) \ // AVL平衡条件
    ((-2 < BalFac(x)) && (BalFac(x) < 2))
tempalte <typename T> class AVL: public BST<T> { // 由BST派生
public: // BST::search()等接口可直接沿用
    BinNodePosi(T) insert(const T &); // 插入重写
    bool remove(const T &); // 删除重写
}

插入节点
插入一个节点可能导致同时有多个失衡节点,g经单旋或双旋调整后复衡,子树高度复原,全树复衡

template <typename T> BinNodePosi(T) AVL<T>::insert(const T & e) {
    BinNodePosi(T) & x = search(e);
    if (x)
        return x;
    x = new BinNode<T>(e, _hot);
    _size++;
    BinNodePosi(T) xx = x; // 若目标不存在,则创建x
    for (BinNodePosi(T) g = x -> parent; g; g = g -> parent) { // 从x的父节点出发逐层向上检查
        if (!AvlBalanced(*g)) { // 若发现g失衡,则通过调整恢复平衡
            FromParentTo(*g) = rotateAt(tallerChild(tallerChild(g)));
            break; // g复衡后,局部子树高度必然复原,调整结束
        } else {
            updateHeight(g); // 否则只需更新高度
        }
    }
    return xx; // 返回新节点,至多需要一次调整
}

删除节点
删除一个节点可能导致同时至多一个失衡节点g,可能是x的父节点_hot
g经单旋调整后复衡,子树高度未必复原,全树仍可能失衡
由于有失衡传播现象,可能需要O(logn)次调整

template <typename T> bool AVL<T>::remove(const T & e) {
    BinNodePosi(T) & x = search(e);
    if (!x)
        return false;
    removeAt(x, _hot);
    _size--; // 若目标存在,则在按BST规则删除后,_hot以上可能失衡
    for (BinNodePosi(T) g = _hot; g; g = g -> parent) { // 从_hot出发逐层向上检查
        if (!AvlBalanced(*g)) // 若发现g失衡,则通过调整恢复平衡
            g = FromParentTo(*g) = rotateAt(tallerChild(tallerChild(g)));
        updateHeight(g); // 更新高度
    } // 可能需要Ω(logn)次调整,无论是否经过调整,全树高度均可能下降
    return true; // 删除成功
}

3+4重构
设g(x)为最低失衡节点,考察三代g ~ p ~ v,按中序遍历次序重命名a < b < c
拥有互不相交的4个子树(可能为空),按中序遍历次序重命名T0 < T1 < T2 < T3

template <typename T> BinNodePosi(T) BST<T>::connect34(BinNodePosi(T) a, BinNodePosi(T) b, BinNodePosi(T) c, BinNodePosi(T) T0, BinNodePosi(T) T1, BinNodePosi(T) T2, BinNodePosi(T) T3) {
    a -> lChild = T0;
    if (T0)
        T0 -> parent = a;
    a -> rChild = T1;
    if (T1)
        T1 -> parent = a;
    updateHeight(a);
    c -> lChild = T2;
    if (T2)
        T2 -> parent = c;
    c -> rChild = T3;
    if (T3)
        T3 -> parent = c;
    updateHeight(c);
    b -> lChild = a;
    a -> parent = b;
    b -> rChild = c;
    c -> parent =b;
    updateHeight(b);
    return b; // 该子树新的根节点
}
template<typename T> BinNodePosi(T) BST<T>::rotateAt(BinNodePosi(T) v) {
    BinNodePosi(T) p = v -> parent;
    BinNodePosi(T) g = p -> parent;
    if (IsLChild(*p)) { // zig
        if (IsLChild(*v)) { // zig-zig
            p -> parent = g -> parent; // 向上联接
            return connect34(v, p, g, v -> lChild, v -> rChild, p -> rChild, g -> rChild);
        } else { // zig-zag
            v -> parent = g -> parent; // 向上联接
            return connect34(p, v, g, p -> lChild, v -> lChild, v -> rChild, g -> rChild);
        }
    } else { // zag
        if(IsRChild(*v)) { // zag-zag
            p->parent = g->parent;
            return connect34(g, p, v, g -> lChild, p -> lChild, v -> lChild, v -> rChild);
        } else { // zag-zig
            v->parent = g->parent;
            return connect34(g, v, p, g -> lChild, v -> lChild, v -> rChild, p -> rChild);
        }
    }
}

AVL树无论查找、插入或删除,最坏情况下的复杂度均为O(logn)
借助高度或平衡因子,需改造元素结构或额外封装
实测复杂度与理论值有差距
单次动态调整后,全树拓扑结构的变化量可能Ω(logn)

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