Java集合中的Set之LinkedHashSet詳解
Set中的LinkedHashSet
LinkedHashSet是Set集合的一個實現(xiàn),具有set集合不重復(fù)的特點,同時具有可預(yù)測的迭代順序,也就是我們插入的順序。
并且linkedHashSet是一個非線程安全的集合。
如果有多個線程同時訪問當(dāng)前l(fā)inkedhashset集合容器,并且有一個線程對當(dāng)前容器中的元素做了修改,那么必須要在外部實現(xiàn)同步保證數(shù)據(jù)的冥等性。
下面我們new一個新的LinkedHashSet容器看一下具體的源碼實現(xiàn)。
并分析師如何保證數(shù)據(jù)的插入順序:
Set<String> set = new LinkedHashSet<>();
跟進(jìn)LinkedHashSet可以得到super一個父類初始化為一個容器為16大小,加載因子為0.75的Map容器。
構(gòu)造一個空連接散列集合
實際創(chuàng)建的是一個LinkedHashMap帶有制定大小和加載因子的容器。
在前面講過一次,map的容器的大小必須是2的冥,那么在講一次如何保證必須是2的冥,通過我們傳入的參數(shù)在構(gòu)建map集合的是通過位運算實現(xiàn):
其中initialCapacity為我們傳入的具體按容器的大小。
上面是我們描述的LinkedHashSet的具體構(gòu)建過程,以及構(gòu)建的具體內(nèi)容。
由于LinkedHashSet是一個哈希表和鏈表的結(jié)合,且是一個雙向鏈表,那么我們來看一下什么是雙向連邊?
雙向鏈表是鏈表的一種,他的每個數(shù)據(jù)節(jié)點都有兩個指針分別指向直接后繼和直接前驅(qū),所以從雙向鏈表的任意一個節(jié)點開始都可以很方便的訪問它的前驅(qū)節(jié)點和后繼節(jié)點。
這是雙向鏈表的優(yōu)點,那么有優(yōu)點就有缺點,缺點是每個節(jié)點都需要保存當(dāng)前節(jié)點的next和prev兩個屬性,這樣才能保證優(yōu)點。
所以需要更多的內(nèi)存開銷,并且刪除和添加也會比較費時間。
下面我們圖示一個雙向兩表的節(jié)點:
多個節(jié)點相互連接,保證了數(shù)據(jù)錄入的順序。
源碼分析
那么我們源碼分析一下具體的錄入詳情:
我們定義一個LinkedHashSet
LinkedHashSet<String> set = new LinkedHashSet<>();
然后set.add();跟一下這個add是走的那個方法:
跟進(jìn)來走的是put的方法:LinkedHashSet.class下的,這個是重寫了超類中put的具體add方法。
他會在新分配的元素在鏈表的末尾插入一條。
進(jìn)來走的還是HashMap的put添加方法,在上面的判斷和計算hash確定位置之后,由于LinkedHashSet重寫了addEntry
在元素的后面添加新的元素。
整個過程就是LinkedHashSet在容器插入數(shù)據(jù)的過程。
此過程主要由LinkedHashSet.class中重寫超類的兩個addEntry和createEntry 實現(xiàn)雙向鏈表的結(jié)構(gòu)。
保證數(shù)據(jù)已我們錄入的順序遍歷輸出。
到此這篇關(guān)于Java集合中的Set之LinkedHashSet詳解的文章就介紹到這了,更多相關(guān)Java中的LinkedHashSet內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Dubbo+zookeeper搭配分布式服務(wù)的過程詳解
Dubbo作為分布式架構(gòu)比較后的框架,同時也是比較容易入手的框架,適合作為分布式的入手框架,下面是簡單的搭建過程,對Dubbo+zookeeper分布式服務(wù)搭建過程感興趣的朋友一起看看吧2022-04-04