8.1 伸展樹(shù)

局部性(Locality):剛被訪問(wèn)過(guò)的數(shù)據(jù),極有可能很快再次被訪問(wèn)
逐層伸展:節(jié)點(diǎn)v一旦被訪問(wèn)迎瞧,隨即轉(zhuǎn)移至根
自上而上邓线,逐層單旋(zig, v->parent; zag, v->parent)顽频,直至v最終被推送至根
最壞情況:旋轉(zhuǎn)次數(shù)呈周期性算術(shù)級(jí)數(shù)演變,每一周期累計(jì)Ω(n2)篷帅,分?jǐn)偊?n)
雙層伸展:向上追溯兩層,反復(fù)考察三代g = parent(p), p = parent(v), v拴泌,根據(jù)其相對(duì)位置魏身,經(jīng)兩次旋轉(zhuǎn)使得v上升兩層,成為子樹(shù)根
zig-zag / zag-zig蚪腐,與AVL樹(shù)雙旋完全等效箭昵,與逐層伸展一致
zig-zig / zag-zag,一旦訪問(wèn)壞節(jié)點(diǎn)回季,對(duì)應(yīng)路徑長(zhǎng)度將隨即減半(折疊效果)家制,最壞情況不致持續(xù)發(fā)生,單次伸展操作分?jǐn)侽(logn)
zig / zag泡一,此時(shí)必有parent(v) == root(T)慰丛,且每輪調(diào)整中至多(在最后)出現(xiàn)一次該情況

接口

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

伸展算法

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)) { // 自下而上,反復(fù)雙層伸展
        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);
    } // 雙層伸展結(jié)束時(shí)必有g(shù) == 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); // 調(diào)用標(biāo)準(zhǔn)BST內(nèi)部接口定位目標(biāo)節(jié)點(diǎn)
    _root = splay(p ? p : _hot); // 無(wú)論成功與否,最后被訪問(wèn)的節(jié)點(diǎn)將伸展至根
    return _root; // 總是返回根節(jié)點(diǎn)
} // 伸展樹(shù)查找操作與常規(guī)BST::search()不同粥烁,可能改變樹(shù)的拓?fù)浣Y(jié)構(gòu)贤笆,不屬于靜態(tài)操作

插入算法

template <typename T> BinNodePosi(T) Splay<T>::insert(const T & e) {
    if (!_root) { // 處理原樹(shù)為空的退化情況
        _size++;
        return _root = new BinNode<T>(e);
    }
    if (e == search(e)->data) // 確認(rèn)目標(biāo)節(jié)點(diǎn)不存在
        return _root;
    _size++;
    BinNodePosi(T) t = _root; // 創(chuàng)建新節(jié)點(diǎn)
    if (_root->data < e) { // 插入新根節(jié)點(diǎn),以t與t->rChild為左右子節(jié)點(diǎn)
        t->parent = _root = new BinNode<T>(e, NULL, t, t->rChild);
        if (HasRChild(*t)) {
            t->rChild->parent = _root;
            t->rChild = NULL;
        }
    } else { // 插入新根節(jié)點(diǎn)讨阻,以t->lChild與t為左右子節(jié)點(diǎn)
        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; // 新節(jié)點(diǎn)必然位于樹(shù)根芥永,返回
}

刪除算法

template <typename T> bool Splay<T>::remove(const T& e) {
    if (!_root || (e != search(e)->data))
        return false; // 若樹(shù)為空或目標(biāo)不存在,則無(wú)法刪除
    BinNodePosi(T) w = _root; // 經(jīng)search()后節(jié)點(diǎn)e已被伸展至樹(shù)根
    if (!HasLChild(*_root)) { // 若無(wú)左子樹(shù)钝吮,則直接刪除
        _root = _root->rChild;
        if (_root)
            _root->parent = NULL;
    } else if (!HasRChild(*_root)) { // 若無(wú)右子樹(shù)埋涧,則直接刪除
        _root = _root->lChild;
        if (_root)
            _root->parent = NULL;
    } else { // 若左右子樹(shù)同時(shí)存在
        BinNodePosi(T) lTree = _root->lChild;
        lTree->parent = NULL;
        _root->lChild = NULL; // ?暫時(shí)刪除左子樹(shù)
        _root = _root->rChild;
        _root->parent = NULL; // 保留右子樹(shù)
        search(w->data); // 以原樹(shù)根為目標(biāo)進(jìn)行查找(必失敯辶伞),至此右子樹(shù)中最小節(jié)點(diǎn)必伸展至根棘催,且因無(wú)相同節(jié)點(diǎn)劲弦,左子樹(shù)必空
        _root->lChild = lTree;
        lTree->parent = _root; // 左子樹(shù)接回原位
    }
    release(w->data);
    release(w);
    _size--; // 釋放節(jié)點(diǎn),更新規(guī)模
    if (_root)
        updateHeight(_root); // 若樹(shù)非空醇坝,則更新樹(shù)根高度
    return true;
}

伸展樹(shù)無(wú)需記錄節(jié)點(diǎn)高度或平衡因子邑跪,編程實(shí)現(xiàn)簡(jiǎn)單易行,優(yōu)于AVL樹(shù)
分?jǐn)倧?fù)雜度O(logn)呼猪,與AVL樹(shù)相當(dāng)
局部性強(qiáng)画畅,緩存命中率極高時(shí)(k << n << m),效率可以更高宋距,任何連續(xù)的m次查找轴踱,可在O(mlogk + nlogn)時(shí)間內(nèi)完成
仍無(wú)法保證單次最壞情況出現(xiàn),不適用于對(duì)效率敏感的場(chǎng)合

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末谚赎,一起剝皮案震驚了整個(gè)濱河市寇僧,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌沸版,老刑警劉巖嘁傀,帶你破解...
    沈念sama閱讀 222,104評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異视粮,居然都是意外死亡细办,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)蕾殴,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)笑撞,“玉大人,你說(shuō)我怎么就攤上這事钓觉≤罘剩” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,697評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵荡灾,是天一觀的道長(zhǎng)瓤狐。 經(jīng)常有香客問(wèn)我,道長(zhǎng)批幌,這世上最難降的妖魔是什么础锐? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,836評(píng)論 1 298
  • 正文 為了忘掉前任,我火速辦了婚禮荧缘,結(jié)果婚禮上皆警,老公的妹妹穿的比我還像新娘。我一直安慰自己截粗,他們只是感情好信姓,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,851評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布鸵隧。 她就那樣靜靜地躺著,像睡著了一般意推。 火紅的嫁衣襯著肌膚如雪豆瘫。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,441評(píng)論 1 310
  • 那天左痢,我揣著相機(jī)與錄音靡羡,去河邊找鬼系洛。 笑死俊性,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的描扯。 我是一名探鬼主播定页,決...
    沈念sama閱讀 40,992評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼绽诚!你這毒婦竟也來(lái)了典徊?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,899評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤恩够,失蹤者是張志新(化名)和其女友劉穎卒落,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體蜂桶,經(jīng)...
    沈念sama閱讀 46,457評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡儡毕,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,529評(píng)論 3 341
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了扑媚。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片腰湾。...
    茶點(diǎn)故事閱讀 40,664評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖疆股,靈堂內(nèi)的尸體忽然破棺而出费坊,到底是詐尸還是另有隱情,我是刑警寧澤旬痹,帶...
    沈念sama閱讀 36,346評(píng)論 5 350
  • 正文 年R本政府宣布附井,位于F島的核電站,受9級(jí)特大地震影響两残,放射性物質(zhì)發(fā)生泄漏羡忘。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,025評(píng)論 3 334
  • 文/蒙蒙 一磕昼、第九天 我趴在偏房一處隱蔽的房頂上張望卷雕。 院中可真熱鬧,春花似錦票从、人聲如沸漫雕。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,511評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)浸间。三九已至太雨,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間魁蒜,已是汗流浹背囊扳。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,611評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留兜看,地道東北人锥咸。 一個(gè)月前我還...
    沈念sama閱讀 49,081評(píng)論 3 377
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像细移,于是被迫代替她去往敵國(guó)和親搏予。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,675評(píng)論 2 359