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

資訊專欄INFORMATION COLUMN

HashMap、HashSet、Hashtable的區(qū)別

zhangxiangliang / 1083人閱讀

摘要:這樣做的目的是提高取對象的效率。在單線程情況下效率較高在的多線程情況下,同步操作能保證程序執(zhí)行的正確性。

突然發(fā)現(xiàn)整理了很多筆記,應該放這里做備用

Hashtable和HashMap

主要區(qū)別:線程安全性,同步(synchronization),以及速度。

HashMap幾乎可以等價于Hashtable,除了HashMap是非synchronized的,并可以接受null。Hashtable是線程安全的,多個線程可以共享一個Hashtable。

HashMap的同步問題可通過Collections的一個靜態(tài)方法得到解決,Map Collections.synchronizedMap(Map m) 返回一個同步的Map。

HashMap的迭代器(Iterator)是fail-fast迭代器,而Hashtable的enumerator迭代器不是fail-fast的。fail-fast結(jié)構(gòu)上更改時(刪除或者插入一個元素),將會拋出ConcurrentModificationException異常。

HashMap不能保證隨著時間的推移Map中的元素次序是不變的。

HashSet和HashMap的區(qū)別

HashSet實現(xiàn)了Set接口,它不允許集合中有重復的值,當我們提到HashSet時,第一件事情就是在將對象存儲在HashSet之前,要先確保對象重寫equals()和hashCode()方法,這樣才能比較對象的值是否相等,以確保set中沒有儲存相等的對象。如果我們沒有重寫這兩個方法,將會使用這個方法的默認實現(xiàn)。

Map中不允許重復的鍵。Map接口有兩個基本的實現(xiàn),HashMap和TreeMap。TreeMap保存了對象的排列次序,而HashMap則不能。HashMap允許鍵和值為null。

HashSet 和 HashMap 之間有很多相似之處,對于 HashSet 而言,系統(tǒng)采用 Hash 算法決定集合元素的存儲位置,這樣可以保證能快速存、取集合元素;對于 HashMap 而言,系統(tǒng) key-value 當成一個整體進行處理,系統(tǒng)總是根據(jù) Hash 算法來計算 key-value 的存儲位置,這樣可以保證能快速存、取 Map 的 key-value 對。

HashMap
public V put(K key, V value) {   
    // 如果 key 為 null,調(diào)用 putForNullKey 方法進行處理  
    if (key == null)   
        return putForNullKey(value);   
    // 根據(jù) key 的 keyCode 計算 Hash 值  
    int hash = hash(key.hashCode());   
    // 搜索指定 hash 值在對應 table 中的索引  
    int i = indexFor(hash, table.length);  
    // 如果 i 索引處的 Entry 不為 null,通過循環(huán)不斷遍歷 e 元素的下一個元素  
    for (Entry e = table[i]; e != null; e = e.next)   
    {   
        Object k;   
        // 找到指定 key 與需要放入的 key 相等(hash 值相同  
        // 通過 equals 比較放回 true)  
        if (e.hash == hash && ((k = e.key) == key   
            || key.equals(k)))   
        {   
            V oldValue = e.value;   
            e.value = value;   
            e.recordAccess(this);   
            return oldValue;   
        }   
    }   
    // 如果 i 索引處的 Entry 為 null,表明此處還沒有 Entry   
    modCount++;   
    // 將 key、value 添加到 i 索引處  
    addEntry(hash, key, value, i);   
    return null;   
}  

Map.Entry,每個 Map.Entry 其實就是一個 key-value 對。

當系統(tǒng)決定存儲 HashMap 中的 key-value 對時,完全沒有考慮 Entry 中的 value,僅僅只是根據(jù) key 來計算并決定每個 Entry 的存儲位置??梢园?Map 集合中的 value 當成 key 的附屬。

indexFor(int h, int length) 方法來計算該對象應該保存在 table 數(shù)組的哪個索引處。

根據(jù)上面 put 方法的源代碼可以看出,當程序試圖將一個 key-value 對放入 HashMap 中時,程序首先根據(jù)該 key 的 hashCode() 返回值決定該 Entry 的存儲位置:如果兩個 Entry 的 key 的 hashCode() 返回值相同,那它們的存儲位置相同。如果這兩個 Entry 的 key 通過 equals 比較返回 true,新添加 Entry 的 value 將覆蓋集合中原有 Entry 的 value,但 key 不會覆蓋。如果這兩個 Entry 的 key 通過 equals 比較返回 false,新添加的 Entry 將與集合中原有 Entry 形成 Entry 鏈,而且新添加的 Entry 位于 Entry 鏈的頭部。

Map提供了一些常用方法,如keySet()、entrySet()等方法,keySet()方法返回值是Map中key值的集合;entrySet()的返回值也是返回一個Set集合,此集合的類型為Map.Entry,接口中有getKey()、getValue方法。

內(nèi)部實現(xiàn)

HashMap實際上是一個“鏈表散列”的數(shù)據(jù)結(jié)構(gòu),即數(shù)組和鏈表的結(jié)構(gòu),但是在jdk1.8里 ,加入了紅黑樹的實現(xiàn),當鏈表的長度大于8時,轉(zhuǎn)換為紅黑樹的結(jié)構(gòu)。

少于8個的時候,Java中HashMap采用了鏈地址法。

通過什么方式來控制map使得Hash碰撞的概率又小,哈希桶數(shù)組(Node[] table)占用空間又少呢?答案就是好的Hash算法和擴容機制。即使負載因子和Hash算法設計的再合理,也免不了會出現(xiàn)拉鏈過長的情況,一旦出現(xiàn)拉鏈過長,則會嚴重影響HashMap的性能。

而當鏈表長度太長(默認超過8)時,鏈表就轉(zhuǎn)換為紅黑樹,利用紅黑樹快速增刪改查的特點提高HashMap的性能。

indexFor方法一般想到的是把hash值對數(shù)組長度取模運算,但運算較大,因此1.7中使用以下方法,&比%具有更高的效率:

static int indexFor(int h, int length) { 
     return h & (length-1);  //第三步 取模運算
}

在JDK1.8的實現(xiàn)中,優(yōu)化了高位運算的算法:(h = k.hashCode()) ^ (h >>> 16)

紅黑樹

紅黑樹(Red Black Tree) 是一種自平衡二叉查找樹。紅黑樹和AVL樹一樣都對插入時間、刪除時間和查找時間提供了最好可能的最壞情況擔保。除了O(log n)的時間之外,紅黑樹的持久版本對每次插入或刪除需要O(log n)的空間。

equals()和hashCode()

兩個obj,如果equals()相等,hashCode()一定相等。
兩個obj,如果hashCode()相等,equals()不一定相等(Hash散列值有沖突的情況,雖然概率很低)。

equals()和hashcode()這兩個方法都是從object類中繼承過來的。equals()是對兩個對象的地址值進行的比較(即比較引用是否相同),hashCode()是一個本地方法,它的實現(xiàn)是根據(jù)本地機器相關的。

hashCode()在new的用途

JVM每new一個Object,它都會將這個Object丟到一個Hash哈希表中去,這樣的話,下次做Object的比較或者取這個對象的時候,它會根據(jù)對象的hashcode再從Hash表中取這個對象。這樣做的目的是提高取對象的效率。

如果不同的對象確產(chǎn)生了相同的hash值,也就是發(fā)生了Hash key相同導致沖突的情況,那么就在這個Hash key的地方產(chǎn)生一個鏈表,將所有產(chǎn)生相同hashcode的對象放到這個單鏈表上去,串在一起。比較兩個對象的時候,首先根據(jù)他們的hashcode去hash表中找他的對象,當兩個對象的hashcode相同,只能根據(jù)Object的equal方法來比較這個對象是否equal。

重寫equals()的原因

Object的equal方法默認是兩個對象的引用的比較,意思就是指向同一內(nèi)存。

但是,String對象中equals方法是判斷值的,而==是地址判斷(因為JDK重寫了)。

我們很大部分時間都是進行兩個對象的比較(而不是比較引用),這個時候Object的equals()方法就不可以了,所以才會有String這些類對equals方法的改寫,依次類推Double、Integer、Math等等這些類都是重寫了equals()方法的,從而進行的是內(nèi)容的比較。

重寫equals()后要重寫hashCode()的理由

java.lnag.Object中對hashCode的約定:

在一個應用程序執(zhí)行期間,如果一個對象的equals方法做比較所用到的信息沒有被修改的話,則對該對象調(diào)用hashCode方法多次,它必須始終如一地返回同一個整數(shù)。

如果兩個對象根據(jù)equals(Object o)方法是相等的,則調(diào)用這兩個對象中任一對象的hashCode方法必須產(chǎn)生相同的整數(shù)結(jié)果。

如果兩個對象根據(jù)equals(Object o)方法是不相等的,則調(diào)用這兩個對象中任一個對象的hashCode方法,不要求產(chǎn)生不同的整數(shù)結(jié)果。但如果能不同,則可能提高散列表的性能。

只要改寫了就會違約,所以要繼續(xù)改寫。

Hashtable與ConcurrentHashMap區(qū)別

ConcurrentHashMap融合了hashtable和hashmap二者的優(yōu)勢。hashmap在單線程情況下效率較高;hashtable在的多線程情況下,同步操作能保證程序執(zhí)行的正確性。

hashtable每次同步執(zhí)行的時候都要鎖住整個結(jié)構(gòu):

ConcurrentHashMap鎖的方式是稍微細粒度的。

更令人驚訝的是ConcurrentHashMap的讀取并發(fā),因為在讀取的大多數(shù)時候都沒有用到鎖定,所以讀取操作幾乎是完全的并發(fā)操作,而寫操作鎖定的粒度又非常細,比起之前又更加快速(這一點在桶更多時表現(xiàn)得更明顯些)。只有在求size等操作時才需要鎖定整個表。

在迭代時,使用弱一致迭代器,不再是拋出 ConcurrentModificationException,在改變時new新的數(shù)據(jù)從而不影響原有的數(shù)據(jù),iterator完成后再將頭指針替換為新的數(shù)據(jù)。

常見問題 HashMap的工作原理,HashMap的get()方法的工作原理

答:“HashMap是基于hashing的原理,我們使用put(key, value)存儲對象到HashMap中,使用get(key)從HashMap中獲取對象。當我們給put()方法傳遞鍵和值時,我們先對鍵調(diào)用hashCode()方法,返回的hashCode用于找到bucket位置來儲存Entry對象?!?/p>

關鍵:HashMap是在bucket中儲存鍵對象和值對象,作為Map.Entry。

當兩個對象的hashcode相同會發(fā)生什么

因為hashcode相同,所以它們的bucket位置相同,發(fā)生沖突,Entry(包含有鍵值對的Map.Entry對象)會存儲在鏈表中。

hashcode相同如何獲取值對象

遍歷鏈表直到找到Entry對象,找到bucket位置之后,會調(diào)用keys.equals()方法去找到鏈表中正確的節(jié)點。

超過了負載因子(load factor)定義的容量

默認的負載因子大小為0.75,將會創(chuàng)建原來HashMap大小的兩倍的bucket數(shù)組,來重新調(diào)整map的大?。╮ehashing)。當多線程的情況下,可能產(chǎn)生條件競爭(race condition),兩個線程都發(fā)現(xiàn)HashMap需要重新調(diào)整大小了,它們會同時試著調(diào)整大小。

先這樣吧

若有錯誤之處請指出,更多地關注煎魚。

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

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

相關文章

  • HashMap,HashSet,Hashtable,Vector,ArrayList 關系

    摘要:繼承的類,泛型為時,注意和其他的類型不同。因為是線程安全簡單來說,是個一維數(shù)組。同樣,指定和,如果中間發(fā)生變化則會拋出異常。最后,可以,然后,使用基類可以實現(xiàn)和的快速賦值。線程安全也是線程安全的,和一樣,連函數(shù)都喪心病狂地同步了。 這么幾個比較常用的但是比較容易混淆的概念同出于 java.util 包。本文僅作幾個類的淺度解析。 (本文基于JDK1.7,源碼來自openjdk1.7。)...

    microcosm1994 評論0 收藏0
  • 這幾道Java集合框架面試題在面試中幾乎必問

    摘要:若遇到哈希沖突,則將沖突的值加到鏈表中即可。之后相比于之前的版本,之后在解決哈希沖突時有了較大的變化,當鏈表長度大于閾值默認為時,將鏈表轉(zhuǎn)化為紅黑樹,以減少搜索時間。有序,唯一紅黑樹自平衡的排序二叉樹。 本文是最最最常見Java面試題總結(jié)系列第三周的文章。主要內(nèi)容: Arraylist 與 LinkedList 異同 ArrayList 與 Vector 區(qū)別 HashMap的底層...

    bigdevil_s 評論0 收藏0
  • 記一次慘痛面試經(jīng)歷

    摘要:把內(nèi)存分成兩種,一種叫做棧內(nèi)存,一種叫做堆內(nèi)存在函數(shù)中定義的一些基本類型的變量和對象的引用變量都是在函數(shù)的棧內(nèi)存中分配。堆內(nèi)存用于存放由創(chuàng)建的對象和數(shù)組。 一次慘痛的阿里技術面 就在昨天,有幸接到了阿里的面試通知,本來我以為自己的簡歷應該不會的到面試的機會了,然而機會卻這么來了,我卻沒有做好準備,被面試官大大一通血虐。因此,我想寫點東西紀念一下這次的經(jīng)歷,也當一次教訓了。其實面試官大大...

    CoorChice 評論0 收藏0
  • 關于java集合框架總結(jié)

    摘要:是哈希表實現(xiàn)的,中的數(shù)據(jù)是無序的,可以放入,但只能放入一個,兩者中的值都不能重復,就如數(shù)據(jù)庫中唯一約束。四和的相同點和區(qū)別相同兩者都是基于哈希表實現(xiàn)的內(nèi)部也都是通過單鏈表解決沖突問題同樣實現(xiàn)了序列化和克隆接口區(qū)別繼承的父類不同。 一.Arraylist與LinkedList有什么區(qū)別? 1、ArrayList是實現(xiàn)了基于動態(tài)數(shù)組的數(shù)據(jù)結(jié)構(gòu),因為地址連續(xù),一旦數(shù)據(jù)存儲好了,查詢操作效率...

    Coding01 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<