HashMap 源碼學(xué)習(xí)

static final int DEFAULT_INITIAL_CAPACITY =16

map初始化的長(zhǎng)度研底,也就是table的長(zhǎng)度

?transient Entry[] table?

table數(shù)據(jù)里存放著每一個(gè)put進(jìn)去的k,v組成的對(duì)象

static final float DEFAULT_LOAD_FACTOR = 0.75f;

int threshold;

初始化賦值默認(rèn)16泼掠,此時(shí)table={}為空

這里了解Entry的結(jié)構(gòu)是相當(dāng)?shù)闹匾?/p>

static class Entry{

final K key;?

V value; ? ? ??

Entry ?next;這里的next也就是表示Entry是一個(gè)鏈,table的每一個(gè)索引存放著一條鏈狀結(jié)構(gòu),同時(shí)對(duì)于相同hash的entry,采用頭插法。

int hash;

/**? ? ? ? * Creates new entry.? ? ? ? */? ? ??

? Entry(int h, K k, V v, Entryn) {

value = v;

next = n;

key = k;

hash = h;

}

}

private void inflateTable(int toSize) {

// Find a power of 2 >= toSize

int capacity = roundUpToPowerOf2(toSize);

threshold = (int) Math.min(capacity * loadFactor, MAXIMUM_CAPACITY + 1);

table = new Entry[capacity];

initHashSeedAsNeeded(capacity);

}

初始化table

public V put(K key, V value) {??

? ? ? if (table == EMPTY_TABLE) {? ? ? ?

?? ? inflateTable(threshold);? ??

? ? }? ? ?

?? if (key == null)? ? ? ??

? ? return putForNullKey(value);? ?

?? ? int hash = hash(key);? ? ?

?? int i = indexFor(hash, table.length);? ? ??

? for (Entrye = table[i]; e != null; e = e.next) {

? ? ? ? Object k;

? ? ? ? if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {

? ? ? ? V oldValue = e.value;

? ? ? ? ?e.value = value;

? ? ? ? ?e.recordAccess(this);

? ? ? ? ?return oldValue;

}

}

modCount++;

addEntry(hash, key, value, i);

return null;

}

put方法的思路大致為:先判斷table是否為空,為空則初始化

然后根據(jù)key值計(jì)算出table里的index盆色,取得對(duì)應(yīng)的entry灰蛙,首先遍歷entry,如果key值已經(jīng)存在那么直接替換值,如果不存在隔躲,在entry的表頭添加最新的key? value.addEntry(hash, key, value, i);

void addEntry(int hash, K key, V value, int bucketIndex) {

if ((size >= threshold) && (null != table[bucketIndex])) {

resize(2 * table.length);

hash = (null != key) ? hash(key) : 0;

bucketIndex = indexFor(hash, table.length);

}

createEntry(hash, key, value, bucketIndex);

}

我們注意到addEntry里當(dāng)table里存放的entry大于初始化時(shí)的capacity便把數(shù)組長(zhǎng)度擴(kuò)大兩倍

void resize(int newCapacity) {

Entry[] oldTable = table;

int oldCapacity = oldTable.length;

if (oldCapacity == MAXIMUM_CAPACITY) {

threshold = Integer.MAX_VALUE;

return;

}

Entry[] newTable = new Entry[newCapacity];

transfer(newTable, initHashSeedAsNeeded(newCapacity));

table = newTable;

threshold = (int)Math.min(newCapacity * loadFactor, MAXIMUM_CAPACITY + 1);

}

數(shù)組擴(kuò)大就要涉及到新數(shù)據(jù)里和老數(shù)據(jù)數(shù)據(jù)復(fù)制的過(guò)程摩梧,所以一般發(fā)生擴(kuò)容的時(shí)候效率上也是受到影響的。

void transfer(Entry[] newTable, boolean rehash) {? ? ? ? int newCapacity = newTable.length;? ? ? ? for (Entrye : table) {? ? ? ? ? ? while(null != e) {? ? ? ? ? ? ? ? Entrynext = e.next;

if (rehash) {

e.hash = null == e.key ? 0 : hash(e.key);

}

int i = indexFor(e.hash, newCapacity);

e.next = newTable[i];

newTable[i] = e;

e = next;

}

}

}

我們注意到對(duì)table里的每一個(gè)index對(duì)應(yīng)的桶bucket復(fù)制之后鏈表的順序是相反了宣旱。仅父。這一點(diǎn)我也不清楚為何要這么處理。

基本了解了hashmap的結(jié)構(gòu)浑吟,其他的方法可以自己去看源碼笙纤。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市组力,隨后出現(xiàn)的幾起案子省容,更是在濱河造成了極大的恐慌,老刑警劉巖燎字,帶你破解...
    沈念sama閱讀 216,470評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件腥椒,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡候衍,警方通過(guò)查閱死者的電腦和手機(jī)寞酿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)脱柱,“玉大人伐弹,你說(shuō)我怎么就攤上這事≌ノ” “怎么了惨好?”我有些...
    開封第一講書人閱讀 162,577評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)随闺。 經(jīng)常有香客問(wèn)我日川,道長(zhǎng),這世上最難降的妖魔是什么矩乐? 我笑而不...
    開封第一講書人閱讀 58,176評(píng)論 1 292
  • 正文 為了忘掉前任龄句,我火速辦了婚禮,結(jié)果婚禮上散罕,老公的妹妹穿的比我還像新娘分歇。我一直安慰自己,他們只是感情好欧漱,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,189評(píng)論 6 388
  • 文/花漫 我一把揭開白布职抡。 她就那樣靜靜地躺著,像睡著了一般误甚。 火紅的嫁衣襯著肌膚如雪缚甩。 梳的紋絲不亂的頭發(fā)上谱净,一...
    開封第一講書人閱讀 51,155評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音擅威,去河邊找鬼壕探。 笑死,一個(gè)胖子當(dāng)著我的面吹牛郊丛,可吹牛的內(nèi)容都是我干的李请。 我是一名探鬼主播,決...
    沈念sama閱讀 40,041評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼宾袜,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼捻艳!你這毒婦竟也來(lái)了驾窟?” 一聲冷哼從身側(cè)響起庆猫,我...
    開封第一講書人閱讀 38,903評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎绅络,沒(méi)想到半個(gè)月后月培,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,319評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡恩急,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,539評(píng)論 2 332
  • 正文 我和宋清朗相戀三年杉畜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片衷恭。...
    茶點(diǎn)故事閱讀 39,703評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡此叠,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出随珠,到底是詐尸還是另有隱情灭袁,我是刑警寧澤,帶...
    沈念sama閱讀 35,417評(píng)論 5 343
  • 正文 年R本政府宣布窗看,位于F島的核電站茸歧,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏显沈。R本人自食惡果不足惜软瞎,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,013評(píng)論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望拉讯。 院中可真熱鬧涤浇,春花似錦、人聲如沸魔慷。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,664評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)盖彭。三九已至纹烹,卻和暖如春页滚,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背铺呵。 一陣腳步聲響...
    開封第一講書人閱讀 32,818評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工裹驰, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人片挂。 一個(gè)月前我還...
    沈念sama閱讀 47,711評(píng)論 2 368
  • 正文 我出身青樓幻林,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親音念。 傳聞我的和親對(duì)象是個(gè)殘疾皇子沪饺,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,601評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容

  • 一整葡、基本數(shù)據(jù)類型 注釋 單行注釋:// 區(qū)域注釋:/* */ 文檔注釋:/** */ 數(shù)值 對(duì)于byte類型而言...
    龍貓小爺閱讀 4,259評(píng)論 0 16
  • 5.1、對(duì)于HashMap需要掌握以下幾點(diǎn) Map的創(chuàng)建:HashMap() 往Map中添加鍵值對(duì):即put(Ob...
    rochuan閱讀 675評(píng)論 0 0
  • 學(xué)習(xí)資料讥脐; 《Java程序性能優(yōu)化》 美團(tuán)點(diǎn)評(píng)技術(shù)團(tuán)隊(duì) Java 8系列之重新認(rèn)識(shí)HashMap 張旭童大佬 面試...
    英勇青銅5閱讀 2,806評(píng)論 3 97
  • 1.什么是HashMap 基于哈希表的Map接口的非同步實(shí)現(xiàn) 此實(shí)現(xiàn)提供所有可選的映射操作遭居,并允許使用null值和...
    蒼賢閱讀 512評(píng)論 0 1
  • 昨天從南昌回來(lái),一直處于高度興奮的狀態(tài)旬渠!當(dāng)越來(lái)越多的品牌俱萍,做來(lái)越多的行業(yè)牛逼人物借助互聯(lián)網(wǎng)來(lái)創(chuàng)業(yè),說(shuō)明微商這個(gè)互聯(lián)...
    若蘭W5243閱讀 262評(píng)論 0 1