java.util系列源碼解讀之TreeMap

TreeMap:基于紅黑樹實現(xiàn)的一個有序的Map實現(xiàn)類.這個有序的維護是通過key實現(xiàn)的Comparable接口或者是在構(gòu)造時傳入的Comparator類來實現(xiàn)它的一個排序規(guī)則的.TreeMap的實現(xiàn)保證了containsKey(),put(),get(),remove()操作的時間復雜度均是log(n)(n是樹上的entry數(shù)).TreeMap是非線程安全類,如果多個線程同時訪問來修改treemap的結(jié)構(gòu)(改變結(jié)構(gòu)是指執(zhí)行了添加或者刪除操作,如果改變一個已經(jīng)存在的key的值這類操作則不算是改變結(jié)構(gòu)),那么必須在外部來保證對這個treemap的同步訪問.
如果要對一個treemap進行同步訪問,我們也可以使用java中提供的同步包裝類來實現(xiàn)同步,例如:

SortedMap m = Collections.synchronizedSortedMap(new TreeMap(...));

下面我們來看看TreeMap中的Entry是一種什么樣的結(jié)構(gòu):

static final class Entry<K,V> implements Map.Entry<K,V> {
    K key; 
    V value;
    Entry<K,V> left = null;  // 左子樹
    Entry<K,V> right = null;  // 右子樹
    Entry<K,V> parent;  // 父節(jié)點
    boolean color = BLACK;  // 本節(jié)點的顏色(紅黑兩種)
    
    Entry(K key, V value, Entry<K,V> parent) {
        this.key = key;
        this.value = value;
        this.parent = parent;
    }
}

類定義

public class TreeMap<K,V>
    extends AbstractMap<K,V>
    implements NavigableMap<K,V>, Cloneable, java.io.Serializable

成員變量

修飾符 變量名 作用
private final Comparator<? super K> comparator 排序比較類
private transient Entry<K,V> root = null 紅黑樹的根節(jié)點
private transient int size = 0 entry數(shù)量
private transient int modCount = 0 記錄改變結(jié)構(gòu)的次數(shù)

方法講解-只講解put和get操作

添加元素 put()

/**
* 1. 首先判斷根元素是否為空,也就是當前put進來的entry是否是第一次操作
* 2. 如果是第一次put,則直接創(chuàng)建entry并賦值給root
* 3. 如果不是第一次put(也就是root不為null),則獲取比較器
* 4. 將新增的key從根節(jié)點開始比較,
*    小于根節(jié)點則繼續(xù)跟當前節(jié)點的左子樹的根節(jié)點比較;
*    大于根節(jié)點則繼續(xù)跟當前節(jié)點的右子樹的根節(jié)點比較;
*    如果等于當前節(jié)點則直接用value替換原來的值并返回原來的值
* 5. 最后如果不是執(zhí)行的替換操作,而是執(zhí)行的插入則要重新調(diào)整樹的結(jié)構(gòu),
*    讓新樹符合紅黑樹的規(guī)則.
**/
public V put(K key, V value) {
    Entry<K,V> t = root;
    if (t == null) {
        compare(key, key); // type (and possibly null) check

        root = new Entry<>(key, value, null);
        size = 1;
        modCount++;
        return null;
    }
    int cmp;
    Entry<K,V> parent;
    // split comparator and comparable paths
    Comparator<? super K> cpr = comparator;
    if (cpr != null) {
        do {
            parent = t;
            cmp = cpr.compare(key, t.key);
            if (cmp < 0)
                t = t.left;
            else if (cmp > 0)
                t = t.right;
            else
                return t.setValue(value);
        } while (t != null);
    }
    else {
        if (key == null)
            throw new NullPointerException();
        Comparable<? super K> k = (Comparable<? super K>) key;
        do {
            parent = t;
            cmp = k.compareTo(t.key);
            if (cmp < 0)
                t = t.left;
            else if (cmp > 0)
                t = t.right;
            else
                return t.setValue(value);
        } while (t != null);
    }
    
    // 遍歷完成,找到新增節(jié)點放置的位置
    Entry<K,V> e = new Entry<>(key, value, parent);
    if (cmp < 0)  // 比父節(jié)點小,放在左子樹
        parent.left = e;
    else   // 大,放在右子樹
        parent.right = e;
    
    // 重新調(diào)整樹使之符合紅黑樹的規(guī)則
    // (此過程比較復雜,需了解紅黑樹算法規(guī)則,在此暫不分析)
    fixAfterInsertion(e);
    size++;
    modCount++;
    return null;
}

獲取元素get()
TreeMap的get()操作就是通過比較循環(huán)獲取左右子樹比較的一個過程,直到找到對應的節(jié)點

public V get(Object key) {
    Entry<K,V> p = getEntry(key);
    return (p==null ? null : p.value);
}

final Entry<K,V> getEntry(Object key) {
    // Offload comparator-based version for sake of performance
    if (comparator != null)
        return getEntryUsingComparator(key);
    if (key == null)
        throw new NullPointerException();
    Comparable<? super K> k = (Comparable<? super K>) key;
    Entry<K,V> p = root;
    while (p != null) {
        int cmp = k.compareTo(p.key);
        if (cmp < 0)
            p = p.left;
        else if (cmp > 0)
            p = p.right;
        else
            return p;
    }
    return null;
}

final Entry<K,V> getEntryUsingComparator(Object key) {
    K k = (K) key;
    Comparator<? super K> cpr = comparator;
    if (cpr != null) {
        Entry<K,V> p = root;
        while (p != null) {
            int cmp = cpr.compare(k, p.key);
            if (cmp < 0)
                p = p.left;
            else if (cmp > 0)
                p = p.right;
            else
                return p;
        }
    }
    return null;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末佛纫,一起剝皮案震驚了整個濱河市首启,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌奴艾,老刑警劉巖顾画,帶你破解...
    沈念sama閱讀 221,548評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件齐媒,死亡現(xiàn)場離奇詭異滨砍,居然都是意外死亡往湿,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評論 3 399
  • 文/潘曉璐 我一進店門惋戏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來领追,“玉大人,你說我怎么就攤上這事响逢∪抟ぃ” “怎么了?”我有些...
    開封第一講書人閱讀 167,990評論 0 360
  • 文/不壞的土叔 我叫張陵舔亭,是天一觀的道長回论。 經(jīng)常有香客問我散罕,道長,這世上最難降的妖魔是什么傀蓉? 我笑而不...
    開封第一講書人閱讀 59,618評論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮职抡,結(jié)果婚禮上葬燎,老公的妹妹穿的比我還像新娘。我一直安慰自己缚甩,他們只是感情好谱净,可當我...
    茶點故事閱讀 68,618評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著擅威,像睡著了一般壕探。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上郊丛,一...
    開封第一講書人閱讀 52,246評論 1 308
  • 那天李请,我揣著相機與錄音,去河邊找鬼厉熟。 笑死导盅,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的揍瑟。 我是一名探鬼主播白翻,決...
    沈念sama閱讀 40,819評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼绢片!你這毒婦竟也來了滤馍?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,725評論 0 276
  • 序言:老撾萬榮一對情侶失蹤底循,失蹤者是張志新(化名)和其女友劉穎巢株,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體此叠,經(jīng)...
    沈念sama閱讀 46,268評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡纯续,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,356評論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了灭袁。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片猬错。...
    茶點故事閱讀 40,488評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖茸歧,靈堂內(nèi)的尸體忽然破棺而出倦炒,到底是詐尸還是另有隱情,我是刑警寧澤软瞎,帶...
    沈念sama閱讀 36,181評論 5 350
  • 正文 年R本政府宣布逢唤,位于F島的核電站拉讯,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏鳖藕。R本人自食惡果不足惜魔慷,卻給世界環(huán)境...
    茶點故事閱讀 41,862評論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望著恩。 院中可真熱鬧院尔,春花似錦、人聲如沸喉誊。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽伍茄。三九已至栋盹,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間敷矫,已是汗流浹背例获。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留沪饺,地道東北人躏敢。 一個月前我還...
    沈念sama閱讀 48,897評論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像整葡,于是被迫代替她去往敵國和親件余。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,500評論 2 359

推薦閱讀更多精彩內(nèi)容

  • 一、基本數(shù)據(jù)類型 注釋 單行注釋:// 區(qū)域注釋:/* */ 文檔注釋:/** */ 數(shù)值 對于byte類型而言...
    龍貓小爺閱讀 4,267評論 0 16
  • 1. Java基礎部分 基礎部分的順序:基本語法俱萍,類相關的語法端壳,內(nèi)部類的語法,繼承相關的語法枪蘑,異常的語法损谦,線程的語...
    子非魚_t_閱讀 31,662評論 18 399
  • java筆記第一天 == 和 equals ==比較的比較的是兩個變量的值是否相等,對于引用型變量表示的是兩個變量...
    jmychou閱讀 1,504評論 0 3
  • 從三月份找實習到現(xiàn)在岳颇,面了一些公司照捡,掛了不少,但最終還是拿到小米话侧、百度栗精、阿里、京東、新浪悲立、CVTE鹿寨、樂視家的研發(fā)崗...
    時芥藍閱讀 42,274評論 11 349
  • 本文轉(zhuǎn)自 https://zhuanlan.zhihu.com/p/21673805 美團點評技術團隊· 3 個月...
    抓兔子的貓閱讀 1,065評論 0 1