摘要:這樣做的目的是提高取對象的效率。在單線程情況下效率較高在的多線程情況下,同步操作能保證程序執(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 對。
HashMappublic 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 (Entrye = 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
摘要:繼承的類,泛型為時,注意和其他的類型不同。因為是線程安全簡單來說,是個一維數(shù)組。同樣,指定和,如果中間發(fā)生變化則會拋出異常。最后,可以,然后,使用基類可以實現(xiàn)和的快速賦值。線程安全也是線程安全的,和一樣,連函數(shù)都喪心病狂地同步了。 這么幾個比較常用的但是比較容易混淆的概念同出于 java.util 包。本文僅作幾個類的淺度解析。 (本文基于JDK1.7,源碼來自openjdk1.7。)...
摘要:若遇到哈希沖突,則將沖突的值加到鏈表中即可。之后相比于之前的版本,之后在解決哈希沖突時有了較大的變化,當鏈表長度大于閾值默認為時,將鏈表轉(zhuǎn)化為紅黑樹,以減少搜索時間。有序,唯一紅黑樹自平衡的排序二叉樹。 本文是最最最常見Java面試題總結(jié)系列第三周的文章。主要內(nèi)容: Arraylist 與 LinkedList 異同 ArrayList 與 Vector 區(qū)別 HashMap的底層...
摘要:把內(nèi)存分成兩種,一種叫做棧內(nèi)存,一種叫做堆內(nèi)存在函數(shù)中定義的一些基本類型的變量和對象的引用變量都是在函數(shù)的棧內(nèi)存中分配。堆內(nèi)存用于存放由創(chuàng)建的對象和數(shù)組。 一次慘痛的阿里技術面 就在昨天,有幸接到了阿里的面試通知,本來我以為自己的簡歷應該不會的到面試的機會了,然而機會卻這么來了,我卻沒有做好準備,被面試官大大一通血虐。因此,我想寫點東西紀念一下這次的經(jīng)歷,也當一次教訓了。其實面試官大大...
摘要:是哈希表實現(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ù)存儲好了,查詢操作效率...
閱讀 651·2021-10-13 09:39
閱讀 1459·2021-09-09 11:53
閱讀 2653·2019-08-29 13:55
閱讀 730·2019-08-28 18:08
閱讀 2599·2019-08-26 13:54
閱讀 2413·2019-08-26 11:44
閱讀 1842·2019-08-26 11:41
閱讀 3791·2019-08-26 10:15