成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

樹 - (二叉查找樹,紅黑樹,B樹)- 紅黑樹

yangrd / 1741人閱讀

摘要:需要執(zhí)行的操作依次是首先,將紅黑樹當(dāng)作一顆二叉查找樹,將該節(jié)點(diǎn)從二叉查找樹中刪除然后,通過旋轉(zhuǎn)和重新著色等一系列來修正該樹,使之重新成為一棵紅黑樹。

雖是讀書筆記,但是如轉(zhuǎn)載請注明出處 http://segmentfault.com/blog/exploring/
.. 拒絕伸手復(fù)制黨

關(guān)于二叉樹的基本知識(shí),可以參見:Java 實(shí)現(xiàn)基本數(shù)據(jù)結(jié)構(gòu) 2(樹)

以下是算法導(dǎo)論第13章的學(xué)習(xí)筆記

紅黑樹

BST的各種操作的時(shí)間復(fù)雜度是依賴于樹的高度,通過使得BST成為紅黑樹,確保每次對BST進(jìn)行插入和刪除之后,樹的高度上限依然是logn.

紅黑樹,本質(zhì)上來說就是一棵二叉查找樹,而且是平衡的查找二叉樹 -- 讓BST效率更優(yōu)

定義

紅黑樹中每個(gè)結(jié)點(diǎn)包含五個(gè)域:color,key,left,rightp。通過對一條從根到葉子的路徑上各個(gè)節(jié)點(diǎn)著色方式的限制,紅黑樹確保沒有一條路徑會(huì)比其他路徑長兩倍。

如果某結(jié)點(diǎn)沒有一個(gè)子結(jié)點(diǎn)或父結(jié)點(diǎn),則該域指向 NIL。

我們把 NIL 視為二叉樹的外結(jié)點(diǎn) (葉子),而帶關(guān)鍵字的結(jié)點(diǎn)視為內(nèi)結(jié)點(diǎn)。

一棵二叉樹如果滿足下面的紅黑性質(zhì),則為一棵紅黑樹:

1) 每個(gè)結(jié)點(diǎn)或是紅的,或是黑的。

2) 根結(jié)點(diǎn)是黑的。

3) 每個(gè)葉結(jié)點(diǎn) (NIL) 是黑的。

4) 如果一個(gè)結(jié)點(diǎn)是紅的,則它的兩個(gè)兒子都是黑的。

5) 對每個(gè)結(jié)點(diǎn),從該結(jié)點(diǎn)到其子孫結(jié)點(diǎn)的所有路徑上包含相同數(shù)目的黑結(jié)點(diǎn)。

采用哨兵來代表 NIL,它的 color 域?yàn)?BLACK,其它域?yàn)槿我庵怠?/p>

從某個(gè)結(jié)點(diǎn) x 出發(fā) (不包括該結(jié)點(diǎn)) 到達(dá)一個(gè)葉結(jié)點(diǎn)的任意一條路徑上,黑色結(jié)點(diǎn)的個(gè)數(shù)稱為該結(jié)點(diǎn)x黑高度,用bh(x) 表示。

引理:一顆有 n 個(gè)內(nèi)結(jié)點(diǎn)的紅黑樹的高度至多為 2lg(n+1)。

操作 旋轉(zhuǎn)

旋轉(zhuǎn)的目的是讓樹保持紅黑樹的特性。

對 x 進(jìn)行左旋,意味著,將 “x 的右孩子” 設(shè)為 “x 的父親節(jié)點(diǎn)”;即,將 x 變成了一個(gè)左節(jié)點(diǎn) (x 成了為 z 的左孩子)!。 因此,左旋中的 “左”,意味著 “被旋轉(zhuǎn)的節(jié)點(diǎn)將變成一個(gè)左節(jié)點(diǎn)”。

對 x 進(jìn)行右旋,意味著,將 “x 的左孩子” 設(shè)為 “x 的父親節(jié)點(diǎn)”;即,將 x 變成了一個(gè)右節(jié)點(diǎn) (x 成了為 y 的右孩子)! 因此,右旋中的 “右”,意味著 “被旋轉(zhuǎn)的節(jié)點(diǎn)將變成一個(gè)右節(jié)點(diǎn)”

        // 左旋x
    public void rotate(TreeNode root, TreeNode x){
        if(x.right != null){
            //處理x的右孩子
            TreeNode y = x.right;
            x.right = y.left;
            if(y.left != null)
                y.left.parent = x;
            // 處理x的父節(jié)點(diǎn)
            y.parent = x.parent ;
            if(x.parent != null){
                // 判斷y鏈接的位置
                if(x.parent.left == x){
                    x.parent.left = y;
                }
                if(x.parent.right == x){
                    x.parent.right = y;
                }
            }else{
                root = y;
            }
            // 鏈接新的父節(jié)點(diǎn)
            x.parent = y;
            y.left = x;
        }
    }

Note: 右旋轉(zhuǎn)的時(shí)候可以把代碼中的left換成right就好了。

插入

關(guān)于插入和刪除整理自July大神的blog和youtube的短視頻
youtube

重溫下RedBlack tree的五條性質(zhì):
1 節(jié)點(diǎn) r or b
2 根 b
3 葉子 b
4 紅色節(jié)點(diǎn)孩子必為黑
5 任意節(jié)點(diǎn)其葉子節(jié)點(diǎn)的路徑包含相同個(gè)數(shù)黑節(jié)點(diǎn)

紅黑樹插入過程的思想是:利用BST的插入方法,尋找待插入元素的位置并插入[所以這一部分可以把BST的直接挪過來]。然后(sth different:) 將待插入元素涂紅色。為了保證紅黑樹的五條性質(zhì),需要調(diào)用輔助程序rbInsertFixup來調(diào)整,對節(jié)點(diǎn)重新著色并旋轉(zhuǎn)。

插入情況(插入的節(jié)點(diǎn)p設(shè)置為紅)有三:
1. 原tree為空樹,所以p設(shè)置為根節(jié)點(diǎn) -- 解決方案:Just 設(shè)置p為黑就可以
2. 插入節(jié)點(diǎn)的父節(jié)點(diǎn)為 -- 無需解決方法,插入后無影響。
3. ** 插入節(jié)點(diǎn)的父節(jié)點(diǎn)為 -- 需要rbInsertFixup
case 1: p節(jié)點(diǎn)和叔節(jié)點(diǎn)都為 -- 解決方案:父+叔 都涂;祖父涂,p = 祖父從新的當(dāng)前結(jié)點(diǎn)重新開始算法
case 2: p節(jié)點(diǎn)為,,且p是父節(jié)點(diǎn)的子 -- 解決方案:p = 父, 左旋p
(case2 實(shí)際上有兩種,看youtube 視頻時(shí)候才發(fā)現(xiàn))
(這兩種情況可以想象成一個(gè)菱形的兩半。只要右子就左旋,左子就右旋)
case 3: p節(jié)點(diǎn)為,,且p是父節(jié)點(diǎn)的子 -- 解決方案:父+叔 都涂, 父節(jié)點(diǎn)涂,祖父涂紅,祖父右旋。
(case3 實(shí)際上也有兩種,這兩種情況可以想象成兩條直線,三角形除了底的兩條邊)

上面三種情況 (Case) 處理問題的核心思路都是:將紅色的節(jié)點(diǎn)移到根節(jié)點(diǎn);然后,將根節(jié)點(diǎn)設(shè)為黑色。

case2 和 case3 的區(qū)分是1. 二者二叉樹的結(jié)構(gòu)不同,菱形和三角 2. 解決方案不同,涂黑or not
最后,把根結(jié)點(diǎn)涂為色,整棵紅黑樹便重新恢復(fù)了平衡。

        //插入
        public RBNode insert(RBNode root, RBNode x){
        RBNode y = this.Nil; // Nil
        RBNode p = root;
        // if the node inserted is null
        if(x == null){
            return root;
        }
        // seek the place where x to be inserted
        while(p!=null){         
            if(x.val > p.val){
                y = p;
                p = p.right;
            }
            if(x.val < p.val){
                y = p;
                p = p.left;
            }
        }
        // insert
        if(y == Nil){
            root = x;
        }
        else
        {
            x.parent = y;
            if(x.val > y.val){
                y.right = x;
            }
            else{
                y.left = x;
            }
        }
        // something different from BST insert 
        x.left = Nil;
        x.right = Nil;
        x.color = 0; // set it red;
        // fixup
        rbInsertFixup(root, x);
        return root;
    }
        public void rbInsertFixup(RBNode root, RBNode x){
        // the fixup occurs when x.partent is red
        while(x.parent.color == 0){
            // 又分為父結(jié)點(diǎn)是祖父結(jié)點(diǎn)的左子還是右子,對于對稱性,我們只要解開一個(gè)方向就可以了
            if(x.parent == x.parent.parent.left){
                RBNode uncle = x.parent.parent.right;
                // case 1 
                if(uncle.color == 0){
                    x.parent.color = 1;
                    uncle.color = 1; 
                    x.parent.parent.color = 0;
                    x = x.parent.parent;
                }

                else
                {
                    // case 2
                    if(x == x.parent.right){
                    {   
                        x = x.parent;
                        this.rotateLeft(root, x);
                    }
                    // case 3
                    {
                        x.parent.color = 1;
                        x.parent.parent.color = 0;
                        this.rotateRight(root, x.parent.parent);
                    }
                }
            else
            {
                    // same as the clause with right and left child
            }
                    }
                }
            root.color = 1;
            }
    }
刪除

摘錄整理自blog
將紅黑樹內(nèi)的某一個(gè)節(jié)點(diǎn)刪除。需要執(zhí)行的操作依次是:
首先,將紅黑樹當(dāng)作一顆二叉查找樹,將該節(jié)點(diǎn)從二叉查找樹中刪除;
然后,通過 "旋轉(zhuǎn)和重新著色" 等一系列來修正該樹,使之重新成為一棵紅黑樹。詳細(xì)描述如下:

第一步:將紅黑樹當(dāng)作一顆二叉查找樹,將節(jié)點(diǎn)刪除。
這和 "刪除常規(guī)二叉查找樹中刪除節(jié)點(diǎn)的方法是一樣的"。分 3 種情況:
① 被刪除節(jié)點(diǎn)沒有兒子,即為葉節(jié)點(diǎn)。那么,直接將該節(jié)點(diǎn)刪除就 OK 了。
② 被刪除節(jié)點(diǎn)只有一個(gè)兒子。那么,直接刪除該節(jié)點(diǎn),并用該節(jié)點(diǎn)的唯一子節(jié)點(diǎn)頂替它的位置。
③ 被刪除節(jié)點(diǎn)有兩個(gè)兒子。那么,先找出它的后繼節(jié)點(diǎn);然后把 “它的后繼節(jié)點(diǎn)的內(nèi)容” 復(fù)制給 “該節(jié)點(diǎn)的內(nèi)容”;之后,刪除 “它的后繼節(jié)點(diǎn)”。在這里,后繼節(jié)點(diǎn)相當(dāng)于替身,在將后繼節(jié)點(diǎn)的內(nèi)容復(fù)制給 "被刪除節(jié)點(diǎn)" 之后,再將后繼節(jié)點(diǎn)刪除。這樣就巧妙的將問題轉(zhuǎn)換為 "刪除后繼節(jié)點(diǎn)" 的情況了,下面就考慮后繼節(jié)點(diǎn)。 在 "被刪除節(jié)點(diǎn)" 有兩個(gè)非空子節(jié)點(diǎn)的情況下,它的后繼節(jié)點(diǎn)不可能是雙子非空。既然 "的后繼節(jié)點(diǎn)" 不可能雙子都非空,就意味著 "該節(jié)點(diǎn)的后繼節(jié)點(diǎn)" 要么沒有兒子,要么只有一個(gè)兒子。若沒有兒子,則按 "情況①" 進(jìn)行處理;若只有一個(gè)兒子,則按 "情況②" 進(jìn)行處理。

第二步:通過 "旋轉(zhuǎn)和重新著色" 等一系列來修正該樹,使之重新成為一棵紅黑樹。
因?yàn)?"第一步" 中刪除節(jié)點(diǎn)之后,可能會(huì)違背紅黑樹的特性。所以需要通過 "旋轉(zhuǎn)和重新著色" 來修正該樹,使之重新成為一棵紅黑樹。

性能
-- BST 紅黑樹 Btree
遍歷 O(n)
插入 O(h) O(h) 4
刪除 O(h) O(h) 1
查詢 O(h) O(h) 2
最小(大) O(h) O(h) 2
后繼(前驅(qū)) O(h) O(h)
旋轉(zhuǎn) - O(1) -
用途

紅黑樹的應(yīng)用比較廣泛,主要是用它來存儲(chǔ)有序的數(shù)據(jù),它的時(shí)間復(fù)雜度是 O(lgn),效率非常之高。
例如,Java 集合中的 TreeSet 和 TreeMap,C++ STL 中的 set、map,以及 Linux 虛擬內(nèi)存的管理,都是通過紅黑樹去實(shí)現(xiàn)的。

和AVL比較

AVL比RBtree更加平衡,但是AVL的插入和刪除會(huì)帶來大量的旋轉(zhuǎn)。
所以如果插入和刪除比較多的情況,應(yīng)該使用RBtree, 如果查詢操作比較多,應(yīng)該使用AVL.

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/64282.html

相關(guān)文章

  • Map集合、散列表、紅黑介紹

    摘要:并且,我們的底層都是紅黑樹來實(shí)現(xiàn)的。紅黑樹是一種平衡二叉樹因此它沒有節(jié)點(diǎn)。 前言 聲明,本文用得是jdk1.8 前面已經(jīng)講了Collection的總覽和剖析List集合: Collection總覽 List集合就這么簡單【源碼剖析】 原本我是打算繼續(xù)將Collection下的Set集合的,結(jié)果看了源碼發(fā)現(xiàn):Set集合實(shí)際上就是HashMap來構(gòu)建的! showImg(https:/...

    2json 評論0 收藏0
  • JDK源碼那些事兒之紅黑基礎(chǔ)上篇

    摘要:用這種范例表示紅黑樹是可能的,但是這會(huì)改變一些性質(zhì)并使算法復(fù)雜。插入會(huì)出現(xiàn)種情況為根節(jié)點(diǎn),即紅黑樹的根節(jié)點(diǎn)。 說到HashMap,就一定要說到紅黑樹,紅黑樹作為一種自平衡二叉查找樹,是一種用途較廣的數(shù)據(jù)結(jié)構(gòu),在jdk1.8中使用紅黑樹提升HashMap的性能,今天就來說一說紅黑樹。 前言 限于篇幅,本文只對紅黑樹的基礎(chǔ)進(jìn)行說明,暫不涉及源碼部分,大部分摘抄自維基百科,這里也貼出對應(yīng)鏈接...

    qylost 評論0 收藏0
  • 數(shù)據(jù)結(jié)構(gòu)與算法:二叉算法

    摘要:因此,根據(jù)題目給出的先序遍歷和中序遍歷,可以畫出二叉樹選參考數(shù)據(jù)結(jié)構(gòu)與算法描述實(shí)現(xiàn)二叉樹算法淺談數(shù)據(jù)結(jié)構(gòu)二叉樹慕課網(wǎng)實(shí)現(xiàn)二叉樹算法前端樹控件騰訊軟件開發(fā)面試題 內(nèi)容銜接上一章 數(shù)據(jù)結(jié)構(gòu)與算法:常見排序算法 內(nèi)容提要 什么是樹   - 為什么使用樹 二叉樹 二叉查找樹 紅黑樹 B、B+樹 堆 伸展樹 樹 可以點(diǎn)擊鏈接感受下筆者用d3.js畫的tree https://codepen...

    Little_XM 評論0 收藏0
  • 紅黑,超強(qiáng)動(dòng)靜圖詳解,簡單易懂

    摘要:寫在前面紅黑樹,對很多童鞋來說,是既熟悉又陌生。每次需要查看紅黑樹內(nèi)容時(shí)都很難以更生動(dòng)形象的方式來理解其內(nèi)容。 寫在前面 紅黑樹,對很多童鞋來說,是既熟悉又陌生。學(xué)校中學(xué)過,只了解大概;工作中不怎么使用,但面試又是重點(diǎn)。每次需要查看紅黑樹內(nèi)容時(shí)都很難以更生動(dòng)形象的方式來理解其內(nèi)容。沒錯(cuò),本文內(nèi)容就是要解決這個(gè)問題,用簡單的語言,搭配靜圖和動(dòng)圖(利用大腦圖形記憶方式),讓你對紅黑樹有更深...

    Scorpion 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<