日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

map怎么转化dto_阿里面试题:为什么Map桶中个数超过8才转为红黑树

發布時間:2023/12/15 编程问答 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 map怎么转化dto_阿里面试题:为什么Map桶中个数超过8才转为红黑树 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

點擊上方“linkoffer”,

選擇關注公眾號高薪職位第一時間送達

這是筆者一個好友面試阿里時,被問及的一個問題,應該不少人看到這個問題都會一面懵逼。因為,大部分的文章都是分析鏈表是怎么轉換成紅黑樹的,但是并沒有說明為什么當鏈表長度為8的時候才做轉換動作。筆者第一反應也是一樣,只能初略的猜測是因為時間和空間的權衡。

要弄明白這個問題,我們首先要明白為什么要轉換,這個問題比較簡單,因為Map中桶的元素初始化是鏈表保存的,其查找性能是O(n),而樹結構能將查找性能提升到O(log(n))。當鏈表長度很小的時候,即使遍歷,速度也非常快,但是當鏈表長度不斷變長,肯定會對查詢性能有一定的影響,所以才需要轉成樹。至于為什么閾值是8,我想,去源碼中找尋答案應該是最可靠的途徑。

8這個閾值定義在HashMap中,如下所示,這段注釋只說明了8是bin(bin就是bucket,即HashMap中hashCode值一樣的元素保存的地方)從鏈表轉成樹的閾值,但是并沒有說明為什么是8:

/**
?*?The?bin?count?threshold?for?using?a?tree?rather?than?list?for?a
?*?bin.??Bins?are?converted?to?trees?when?adding?an?element?to?a
?*?bin?with?at?least?this?many?nodes.?The?value?must?be?greater
?*?than?2?and?should?be?at?least?8?to?mesh?with?assumptions?in
?*?tree?removal?about?conversion?back?to?plain?bins?upon?shrinkage.
?*/
static?final?int?TREEIFY_THRESHOLD?=?8;

我們繼續往下看,在HashMap中有一段Implementation notes,筆者摘錄了幾段重要的描述,第一段如下所示,大概含義是當bin變得很大的時候,就會被轉換成TreeNodes中的bin,其結構和TreeMap相似,也就是紅黑樹:

This?map?usually?acts?as?a?binned?(bucketed)?hash?table,?but
when?bins?get?too?large,?they?are?transformed?into?bins?of?TreeNodes,
each?structured?similarly?to?those?in?java.util.TreeMap

繼續往下看,TreeNodes占用空間是普通Nodes的兩倍,所以只有當bin包含足夠多的節點時才會轉成TreeNodes,而是否足夠多就是由TREEIFY_THRESHOLD的值決定的。當bin中節點數變少時,又會轉成普通的bin。并且我們查看源碼的時候發現,鏈表長度達到8就轉成紅黑樹,當長度降到6就轉成普通bin。

這樣就解析了為什么不是一開始就將其轉換為TreeNodes,而是需要一定節點數才轉為TreeNodes,說白了就是trade-off,空間和時間的權衡:

Because?TreeNodes?are?about?twice?the?size?of?regular?nodes,?we
use?them?only?when?bins?contain?enough?nodes?to?warrant?use
(see?TREEIFY_THRESHOLD).?And?when?they?become?too?small?(due?to
removal?or?resizing)?they?are?converted?back?to?plain?bins.??In
usages?with?well-distributed?user?hashCodes,?tree?bins?are
rarely?used.??Ideally,?under?random?hashCodes,?the?frequency?of
nodes?in?bins?follows?a?Poisson?distribution
(http://en.wikipedia.org/wiki/Poisson_distribution)?with?a
parameter?of?about?0.5?on?average?for?the?default?resizing
threshold?of?0.75,?although?with?a?large?variance?because?of
resizing?granularity.?Ignoring?variance,?the?expected
occurrences?of?list?size?k?are?(exp(-0.5)*pow(0.5,?k)/factorial(k)).?
The?first?values?are:
0:????0.60653066
1:????0.30326533
2:????0.07581633
3:????0.01263606
4:????0.00157952
5:????0.00015795
6:????0.00001316
7:????0.00000094
8:????0.00000006
more:?less?than?1?in?ten?million

這段內容還說到:當hashCode離散性很好的時候,樹型bin用到的概率非常小,因為數據均勻分布在每個bin中,幾乎不會有bin中鏈表長度會達到閾值。但是在隨機hashCode下,離散性可能會變差,然而JDK又不能阻止用戶實現這種不好的hash算法,因此就可能導致不均勻的數據分布。不過理想情況下隨機hashCode算法下所有bin中節點的分布頻率會遵循泊松分布,我們可以看到,一個bin中鏈表長度達到8個元素的概率為0.00000006,幾乎是不可能事件。所以,之所以選擇8,不是拍拍屁股決定的,而是根據概率統計決定的。由此可見,發展30年的Java每一項改動和優化都是非常嚴謹和科學的。

  • 畫外音

筆者通過搜索引擎搜索這個問題,發現很多下面這個答案(猜測也是相互轉發):

紅黑樹的平均查找長度是log(n),如果長度為8,平均查找長度為log(8)=3,鏈表的平均查找長度為n/2,當長度為8時,平均查找長度為8/2=4,這才有轉換成樹的必要;鏈表長度如果是小于等于6,6/2=3,而log(6)=2.6,雖然速度也很快的,但是轉化為樹結構和生成樹的時間并不會太短。

筆者認為這個答案不夠嚴謹:3相比4有轉換的必要,而2.6相比3就沒有轉換的必要?起碼我不敢茍同這個觀點。

跪求簡歷!上海/北京周末

螞蟻、阿里專場招聘,要求:3年以上java經驗,有高可用分布式經驗,扎實的技術基礎,熟悉設計模式,掌握主流框架。有互聯網,銀行,支付,金融等背景優先!

簡歷投遞:1611972891@qq.com,咨詢微信13250325307。

總結

以上是生活随笔為你收集整理的map怎么转化dto_阿里面试题:为什么Map桶中个数超过8才转为红黑树的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。