欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

Java8中HashMap有必要來看下探討下了-創(chuàng)新互聯(lián)

Java8中HashMap有必要來看下探討下了

Java8中HashMap有必要來看下探討下了

為邗江等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計制作服務(wù),及邗江網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為成都網(wǎng)站設(shè)計、做網(wǎng)站、邗江網(wǎng)站設(shè)計,以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達到每一位用戶的要求,就會得到認可,從而選擇與我們長期合作。這樣,我們也可以走得更遠!

HashMap是Java開發(fā)當中使用得非常多的一種數(shù)據(jù)結(jié)構(gòu),因為其可以快速的定位到需要查找到數(shù)據(jù),其最快的速度可以達到O(1),最差的時候也可以達到O(n)。本文以Java8中的HashMap做為分析原型,因為不同的JDK版本中的HashMap,可能存在著底層實現(xiàn)上的不一樣。

HashMap是通過數(shù)組存儲所有的數(shù)據(jù),每個元素所存放數(shù)組的下標,是根據(jù)該存儲元素的key的Hash值與該數(shù)組的長度減去1做與運算,如下所示:

index?=?(length_of_array?-?1)?&?hash_of_the_key;

數(shù)組中存放元素的數(shù)據(jù)結(jié)構(gòu)使用了Node和TreeNode兩種數(shù)據(jù)結(jié)構(gòu),在單個Hash值對應(yīng)的存儲元素小于8個時,默認值為Node的單向鏈表形式存儲,當單個Hash值存儲的元素大于8個時,其會使用TreeNode的數(shù)據(jù)結(jié)構(gòu)存儲。

因為在單個Hash值對應(yīng)的元素小于等于8個時,其查詢時間最差為O(8),但是當單個Hash值對應(yīng)的元素大于8個時,再通過Node的單向鏈表的方式進行查詢,速度上就會變得更慢了;這個時候HashMap就會將Node的普通節(jié)點轉(zhuǎn)為TreeNode(紅黑樹)進行存儲,這是由于TreeNode占用的空間大小約為常規(guī)節(jié)點的兩倍,但是其查詢速度可以得到保證,這個是通過空間換時間了。當TreeNode中包括的元素變得比較少時,為了存儲空間的占用,也會轉(zhuǎn)換為Node節(jié)點單向鏈表的方式實現(xiàn),它們之間可以互相轉(zhuǎn)換的。

Node:

static?class?Node<K,V>?implements?Map.Entry<K,V>?{
?final?int?hash;
?final?K?key;
?V?value;?Node<K,V>?next;?Node(int?hash,?K?key,?V?value,?Node<K,V>?next)?{
?this.hash?=?hash;
?this.key?=?key;
?this.value?=?value;
?this.next?=?next;
?}
?......
}

可以看到每個Node中包括了4個屬性,分別為:

hash值:當前Node的Hash值
key:當前Node的keyvalue:當前Node的valuenext:表示指向下一個Node的指針,相同hash值的Node,通過next進行遍歷查找

TreeNode:

static?final?class?TreeNode<K,V>?extends?LinkedHashMap.Entry<K,V>?{?TreeNode<K,V>?parent;?//?red-black?tree?links
?TreeNode<K,V>?left;?TreeNode<K,V>?right;?TreeNode<K,V>?prev;?//?needed?to?unlink?next?upon?deletion
?boolean?red;?TreeNode(int?hash,?K?key,?V?val,?Node<K,V>?next)?{?super(hash,?key,?val,?next);
?}
?......
}

可以看到TreeNode使用的是紅黑樹(Red Black Tree)的數(shù)據(jù)結(jié)構(gòu),紅黑樹是一種自平衡二叉查找樹,在進行插入和刪除操作時通過特定操作保持二叉查找樹的平衡,從而獲得較高的查找性能,即使在最壞情況運行時間也是非常良好的,并且在實踐中是非常高效的,它可以在O(log n)時間內(nèi)做查找、插入和刪除等操作,這里的n 是樹中元素的數(shù)目。

以下是一張關(guān)于HashMap存儲結(jié)構(gòu)的示意圖:

Java8中HashMap有必要來看下探討下了

寫入數(shù)據(jù)(一切皆在注釋中)

其方法如下:

//寫入數(shù)據(jù)public?V?put(K?key,?V?value)?{?//首先根據(jù)hash方法,獲取對應(yīng)key的hash值,計算方法見后面
?return?putVal(hash(key),?key,?value,?false,?true);
}final?V?putVal(int?hash,?K?key,?V?value,?boolean?onlyIfAbsent,boolean?evict)?{
?Node<K,V>[]?tab;?Node<K,V>?p;?int?n,?i;?//判斷用戶存放元素的數(shù)組是否為空
?if?((tab?=?table)?==?null?||?(n?=?tab.length)?==?0)?//為空則進行初使化,并將初使化后的數(shù)組賦值給變量tab,數(shù)組的長值賦值給變量n
?n?=?(tab?=?resize()).length;?//判斷根據(jù)hash值與數(shù)組長度減1求與得到的下標,
?//從數(shù)組中獲取元素并將其賦值給變量p(后續(xù)該變量p可以繼續(xù)使用),并判斷該元素是否存在
?if?((p?=?tab[i?=?(n?-?1)?&?hash])?==?null)?//如果不存在則創(chuàng)建一個新的節(jié)點,并將其放到數(shù)組對應(yīng)的下標中
?tab[i]?=?newNode(hash,?key,?value,?null);?else?{//根據(jù)數(shù)組的下標取到了元素,并且該元素p且不為空,下面要判斷p元素的類型是Node還是TreeNode
?Node<K,V>?e;?K?k;?//判斷該數(shù)組對應(yīng)下標取到的第一值是不是與正在存入值的hash值相同、
?//key相等(可能是對象,也可能是字符串),如果相等,則將取第一個值賦值給變量e
?if?(p.hash?==?hash?&&
?((k?=?p.key)?==?key?||?(key?!=?null?&&?key.equals(k))))
?e?=?p;?//判斷取的對象是不是TreeNode,如果是則執(zhí)行TreeNode的put方法
?else?if?(p?instanceof?TreeNode)
?e?=?((TreeNode<K,V>)p).putTreeVal(this,?tab,?hash,?key,?value);?else?{//是普通的Node節(jié)點,
?//根據(jù)next屬性對元素p執(zhí)行單向鏈表的遍歷
?for?(int?binCount?=?0;?;?++binCount)?{?//如果被遍歷的元素最后的next為空,表示后面沒有節(jié)點了,則將新節(jié)點與當前節(jié)點的next屬性建立關(guān)系
?if?((e?=?p.next)?==?null)?{?//做為當前節(jié)點的后面的一個節(jié)點
?p.next?=?newNode(hash,?key,?value,?null);?//判斷當前節(jié)點的單向鏈接的數(shù)量(8個)是不是已經(jīng)達到了需要將其轉(zhuǎn)換為TreeNode了
?if?(binCount?>=?TREEIFY_THRESHOLD?-?1)?//?-1?for?1st
?//如果是則將當前數(shù)組下標對應(yīng)的元素轉(zhuǎn)換為TreeNode
?treeifyBin(tab,?hash);?break;
?}?//判斷待插入的元素的hash值與key是否與單向鏈表中的某個元素的hash值與key是相同的,如果是則退出
?if?(e.hash?==?hash?&&
?((k?=?e.key)?==?key?||?(key?!=?null?&&?key.equals(k))))?break;
?p?=?e;
?}
?}?//判斷是否找到了與待插入元素的hash值與key值都相同的元素
?if?(e?!=?null)?{?//?existing?mapping?for?key
?V?oldValue?=?e.value;?//判斷是否要將舊值替換為新值
?if?(!onlyIfAbsent?||?oldValue?==?null)?//滿足于未指定不替換或舊值為空的情況,執(zhí)行將舊值替換為新值
?e.value?=?value;
?afterNodeAccess(e);?return?oldValue;
?}
?}
?++modCount;?if?(++size?>?threshold)
?resize();
?afterNodeInsertion(evict);?return?null;
}

Hash值的計算方法:

//?計算指定key的hash值,原理是將key的hash?code與hash?code無符號向右移16位的值,執(zhí)行異或運算。//?在Java中整型為4個字節(jié)32位,無符號向右移16位,表示將高16位移到低16位上,然后再執(zhí)行異或運行,也//?就是將hash?code的高16位與低16位進行異或運行。//?小于等于65535的數(shù),其高16位全部都為0,因而將小于等于65535的值向右無符號移16位,則該數(shù)就變成了//?32位都是0,由于任何數(shù)與0進行異或都等于本身,因而hash?code小于等于65535的key,其得到的hash值//?就等于其本身的hash?code。static?final?int?hash(Object?key)?{?int?h;?return?(key?==?null)???0?:?(h?=?key.hashCode())?^?(h?>>>?16);
}

計算邏輯如下圖所示:

Java8中HashMap有必要來看下探討下了

讀取數(shù)據(jù)(一切皆在注釋中)

?public?V?get(Object?key)?{
?Node<K,V>?e;?//根據(jù)Key獲取元素
?if?((e?=?getNode(hash(key),?key))?==?null)?return?null;?if?(accessOrder)
?afterNodeAccess(e);?return?e.value;
?}?final?Node<K,V>?getNode(int?hash,?Object?key)?{
?Node<K,V>[]?tab;?Node<K,V>?first,?e;?int?n;?K?k;?//if語句的第一個判斷條件
?if?((tab?=?table)?!=?null?//將數(shù)組賦值給變量tab,將判斷是否為null
?&&?(n?=?tab.length)?>?0?//將數(shù)組的長值賦值給變量n
?&&?(first?=?tab[(n?-?1)?&?hash])?!=?null)?{//判斷根據(jù)hash和數(shù)組長度減1的與運算,計算出來的的數(shù)組下標的第一個元素是不是為空
?//判斷第一個元素是否要找的元素,大部份情況下只要hash值太集中,或者元素不是很多,第一個元素往往都是需要的最終元素
?if?(first.hash?==?hash?&&?//?always?check?first?node
?((k?=?first.key)?==?key?||?(key?!=?null?&&?key.equals(k))))?//第一個元素就是要找的元素,因為hash值和key都相等,直接返回
?return?first;?if?((e?=?first.next)?!=?null)?{//如果第一元素不是要找到的元,則判斷其next指向是否還有元素
?//有元素,判斷其是否是TreeNode
?if?(first?instanceof?TreeNode)?//是TreeNode則根據(jù)TreeNode的方式獲取數(shù)據(jù)
?return?((TreeNode<K,V>)first).getTreeNode(hash,?key);
?do?{//是Node單向鏈表,則通過next循環(huán)匹配,找到就退出,否則直到匹配完最后一個元素才退出
?if?(e.hash?==?hash?&&
?((k?=?e.key)?==?key?||?(key?!=?null?&&?key.equals(k))))?return?e;
?}?while?((e?=?e.next)?!=?null);
?}
?}?//沒有找到則返回null
?return?null;
?}

喜歡這篇文章的話,可以給作者點個喜歡,點下關(guān)注,每天都會分享Java相關(guān)文章!

記得一定要關(guān)注我哦,會不定時的福利贈送,包括整理的面試題,學(xué)習(xí)資料,源碼等~~

另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。

文章名稱:Java8中HashMap有必要來看下探討下了-創(chuàng)新互聯(lián)
轉(zhuǎn)載源于:http://aaarwkj.com/article42/cojdhc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供電子商務(wù)、定制網(wǎng)站、微信公眾號、品牌網(wǎng)站制作、品牌網(wǎng)站建設(shè)、定制開發(fā)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站建設(shè)
日本黄色免费在线观看网站| 91精品一区二区三区91人妻| 日本亚洲欧洲一区二区| 国产精品传媒成人免费| av在线免费观看美日韩| 男女做爰高清无遮挡免费| 亚洲国产高清国产拍精品| 高清av网站大全网站| 国产欧美日韩午夜激情| 日韩精品一区高清视频| 免费看真人性生活视频 | 四虎官网免费在线观看| 精品欧美高清免费视频| 亚洲欧美日韩性生活视频| 亚洲家庭伦理在线观看| 亚洲毛片一区在线播放| 亚洲永久精品天码野外| 亚洲欧美成人综合网站| 久久精品欧美日韩视频| 六月综合激情丁香婷婷色| 白虎亚洲福利精品一区| 精品视频一区二区三区在线观看 | 男人天堂插插综合搜索| 亚洲国产精品视频自拍| 91蜜臀视频在线播放| 国产视频三级在线观看| 日韩欧美一区二区大片| 偷怕自拍在线免费观看| 在线观看免费国产b片| 国产成人久久精品二区三区| 成人午夜激情福利动态| 懂色av中文一区二区| 亚洲黄色片成年人免费观看| 国产真人作爱视频免费| 国产传媒视频网站在线观看| 精品自拍一区在线观看| 人妻中字幕出轨中文字幕| 欧美高清一区二区三区不卡| 日本免费熟女一区二区| 欧美日韩黄色人人小视频| 日本不卡一二三区在线观看|