高效Java第九條覆蓋equals時總要覆蓋hashCode

在每個覆蓋了equals方法的類中,也必須覆蓋hashCode方法。
否則會導致該類無法與基于散列的集合一起正常運作。

hashCode約定

  • 在應用程序的執行期間,只要對象的equals方法所用到的信息沒有被修改,那么對著同一個對象調用多次,hashCode方法都必須始終如一地返回同一個整數。
  • 如果兩個對象equals(Object)方法比較是相等的,那么調用這兩個對象的hashCode方法必須產生同樣的整數結果。
  • 如果兩個對象equals(Object)比較是不相等的,那么調用這兩個對象中的hashCode方法,則不一定要產生不同的整數結果。但是給不相等的對象產生截然不同的整數結果,有可能提高散列表的性能。

沒有覆蓋hashCode違反第二條約定

因沒有覆蓋hashCode而違反的關鍵約定是第二條:相等的對象必須具有相等的散列碼。

PhoneNumber例子——沒有覆蓋hashCode

該類無法與HashMap一起正常工作:

m.get(new PhoneNumber(408, 867, 5309))返回null
由于PhoneNumber類沒有覆蓋hashCode方法,從而導致兩個相等的實例具有不相等的散列碼,違反了hashCode的約定。因此,put方法把電話號碼對象存放在一個散列桶中,get方法卻在另一個散列桶中查找這個電話號碼。即使這兩個實例正好被放到同一個散列桶中,get方法也必定會返回null,因為HahsMsap有一項優化,可以將每個項相關聯的散列碼緩存起來,如果散列碼不匹配,也不必檢驗對象的等同性。

PhoneNumber類提供一個適當的hashCode方法。
下面的hashCode方法是錯誤的:

雖然上面的hashCode方法確保了相等的對象總是具有同樣的散列碼。但是它使得每個對象都具有同樣的散列碼。因此,每個對象都被映射到同一個散列桶中,使散列表退化為鏈表。它使得本該線性時間運行的程序變成了以平方級時間在運行。對于規模很大的散列表而言,這關系到散列表能否正常工作。

如何編寫好的散列函數

一個好的散列函數傾向于為不相等的對象產生不相等的散列碼。
散列函數應該把集合中不相等的實例均勻地分布到所有可能的散列值上。

1.把某個非零的常數值,比如17,保存在一個名為resultint類型的變量中。
2.對于對象中每個關鍵域f(指equals方法涉及的每個域),完成以下步驟:
a。為該域計算int類型的散列碼c:
i.如果該域是boolean類型,則計算(f?1:0)。
ii.如果該域是bytecharshort或者int類型,則計算(int)f
iii.如果該域是long類型,則計算(int)(f ^ (f >>> 32))
iv.如果該域是float類型,則計算Float.floatToIntBits(f)
v.如果該域是double類型,則計算Double.doubleToLongBits(f),然后按照步驟2.a.iii,為得到的long類型值計算散列值。
vi.如果該域是一個對象引用,并且該類的equals方法通過遞歸地調用equals方法來比較這個域,則同樣為這個域遞歸地調用hashCode。如果需要更復雜的比較,則為這個域計算一個范式,然后針對這個范式調用hashCode。如果這個域的值為null,則返回0(或者其他某個常數,但通常是0)。

vii.如果該域是一個數組,則要把每一個元素當做單獨的域來處理。遞歸地應用上述規則,對每個重要的元素計算一個散列碼,然后根據步驟2.b中的做法把這些散列值組合起來。如果數組域中的每個元素都很重要,建議使用Arrays.hashCode方法。
b.把步驟2.a中計算得到的散列碼c合并到result中:
result = 31 * result + c;
3.返回result

4.寫完了hashCode方法之后,問問自己“相等的實例是否都具有相等的散列碼”。建議編寫單元測試。

計算散列碼可以把冗余域排除在外。如果一個域的值可以根據參與計算的其他域的值計算出來,則可以把這樣的域排除在外。必須排除equals比較計算中沒有用到的任何域,否則很有可能違反hashCode約定的第二條。

值17是任選的。
步驟2.b中的乘法部分使得散列值依賴于域的順序,如果一個類中包含多個相似的域,這樣的乘法運算就會產生一個更好的散列函數。String的散列函數省略了乘法部分,那么只是字母順序不同的所有字符串就會有相同的散列碼。
選擇31是因為它是一個奇素數。如果乘數是偶數,并且乘法溢出的話,信息就會丟失,因為與2相乘等價于移位運算。使用素數的好處并不很明顯,但習慣上都使用素數來計算散列結果。31可以利用移位和減法來代替乘法,可以得到更好的性能:31 * i == (i << 5) -i。JVM可以自動完成這種優化。

PhoneNumber編寫好的hashCode

PhoneNumber類的hashCode

緩存散列碼

不可變類如果計算散列碼的開銷比較大,就應該考慮把散列碼緩存在對象內部,而不是每次請求的時候都重新計算散列碼。
如果類的大多數對象都會被用做散列鍵,就應該在創建實例的時候計算散列碼。否則,可以選擇“延遲初始化”散列碼,一直到hashCode被第一次調用的時候才初始化。
PhoneNumber類的對象有可能會經常被作為散列鍵:

不要排除對象的關鍵部分

不要試圖從散列碼計算中排除掉一個對象的關鍵部分來提高性能。即使這樣得到的散列函數運行起來更快,但是它的效果不見得會好,可能會導致散列表慢到根本無法使用。
在實踐中,散列函數可能面臨大量的實例,在選擇忽略的區域之中,這些實例區別非常大。散列函數會把所有這些實例映射到極少數的散列碼上,基于散列的集合將會顯示出平方級的性能指標。
JDK2之前的String散列函數至多只檢查16個字符,從第一個字符開始,在整個字符串中均勻選取。對于大型集合,該散列函數表現出了病態行為。

不要規定散列函數的細節

Java平臺類庫中的許多類,StringIntegerDate等都可以把hashCode方法返回的確切值規定為該實例值的一個函數。但是這并不是一個好注意,這會嚴格地限制了在將來的版本中改進散列函數的能力。如果沒有規定散列函數的細節,那么當你發現了它的內部缺陷時,就可以在后面的發行版本中修正它,確信沒有任何客戶端會依賴于散列函數返回的確切值。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,885評論 6 541
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,312評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,993評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,667評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,410評論 6 411
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,778評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,775評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,955評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,521評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,266評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,468評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,998評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,696評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,095評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,385評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,193評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,431評論 2 378

推薦閱讀更多精彩內容