hashCode()
方法的對象若是沒有覆蓋 hashCode()
方法,那麼哈希值爲底層 JDK C++ 源碼實現,實例每次調用hashcode()
方法,只有第一次計算哈希值,以後哈希值會存儲在對象頭的 標記字(MarkWord) 中。git
若是進入各類鎖狀態,那麼會緩存在其餘地方,通常是獲取鎖的線程裏面存儲,恢復無鎖(即釋放鎖)會改回原有的哈希值。github
對應源碼synchronizer.cpp
:編程
//若是是無鎖狀態 if (mark.is_neutral()) { hash = mark.hash(); //若是hash不等於0,證實計算過,直接返回 if (hash != 0) { return hash; } //不然,計算hash值 hash = get_next_hash(self, obj); // get a new hash //拷貝到Header中記錄 temp = mark.copy_set_hash(hash); test = obj->cas_set_mark(temp, mark); //可能有併發,並且不一樣默認哈希值計算方法,可能每次哈希值不同,只有 CAS 成功的纔是最後的哈希值 //默認的哈希值計算,不論計算多少次,都不會變 if (test == mark) { return hash; } } else if (mark.has_monitor()) { //若是是有 monitor 鎖狀態(重量級鎖),則獲取其 monitor,哈希值會記錄在monitor的頭部 monitor = mark.monitor(); temp = monitor->header(); assert(temp.is_neutral(), "invariant: header=" INTPTR_FORMAT, temp.value()); hash = temp.hash(); if (hash != 0) { OrderAccess::loadload_for_IRIW(); if (monitor->is_being_async_deflated()) { monitor->install_displaced_markword_in_object(obj); continue; } return hash; } } else if (self->is_lock_owned((address)mark.locker())) { // 若是是輕量級鎖狀態,獲取輕量鎖,其中也記錄着以前計算的哈希值 temp = mark.displaced_mark_helper(); assert(temp.is_neutral(), "invariant: header=" INTPTR_FORMAT, temp.value()); hash = temp.hash(); if (hash != 0) { // if it has a hash, just return it return hash; } }
hashCode()
方法的對象對於已經覆蓋hashCode()
方法的對象,則每次都會從新調用hashCode()
方法從新計算哈希值。緩存
微信搜索「個人編程喵」關注公衆號,每日一刷,輕鬆提高技術,斬獲各類offer:微信