簡(jiǎn)述Hibernate兩種方法的實(shí)現(xiàn)
Hibernate有很多方法,每一種方法都有自己實(shí)現(xiàn)的過(guò)程,這里詳述Hibernate兩種方法的實(shí)現(xiàn)Hibernate equals()和Hibernate hashCode()。
實(shí)現(xiàn)Hibernate equals()和hashCode()如果你有如下需求,你必須重載 equals() 和 hashCode()方法:
想把持久類的實(shí)例放入Set中(當(dāng)表示多值關(guān)聯(lián)時(shí),推薦這么做)
想重用脫管實(shí)例
Hibernate保證,僅在特定會(huì)話范圍內(nèi),持久化標(biāo)識(shí)(數(shù)據(jù)庫(kù)的行)和Java標(biāo)識(shí)是等價(jià)的。因此,一旦 我們混合了從不同會(huì)話中獲取的實(shí)例,如果希望Set有明確的語(yǔ)義,就必 須實(shí)現(xiàn)equals() 和hashCode()。
實(shí)現(xiàn)Hibernate equals()/HibernatehashCode()最顯而易見(jiàn)的方法是比較兩個(gè)對(duì)象標(biāo)識(shí)符的值。如果值相同,則兩個(gè)對(duì)象對(duì)應(yīng)于數(shù)據(jù)庫(kù)的同一行,因此它們是相等的(如果都被添加到 Set,則在Set中只有一個(gè)元素)。
不幸的是,對(duì)生成的標(biāo)識(shí)不能 使用這種方法。Hibernate僅對(duì)那些持久化對(duì)象賦標(biāo)識(shí)值,一個(gè)新創(chuàng)建的實(shí)例將不會(huì)有任何標(biāo)識(shí)值。此外, 如果一個(gè)實(shí)例沒(méi)有被保存(unsaved),并且它當(dāng)前正在一個(gè)Set中,保存它將會(huì)給這個(gè)對(duì)象 賦一個(gè)標(biāo)識(shí)值。
如果Hibernate equals() 和Hibernate hashCode()是基于標(biāo)識(shí)值 實(shí)現(xiàn)的,則其哈希碼將會(huì)改變,這違反了Set的契約。建議去Hibernate的站點(diǎn)閱讀關(guān)于這個(gè) 問(wèn)題的全部討論。
注意:這不是Hibernate的問(wèn)題,而是一般的Java對(duì)象標(biāo)識(shí)和Java對(duì)象等價(jià)的語(yǔ)義問(wèn)題。
我們建議使用業(yè)務(wù)鍵值相等(Business key equality)來(lái)實(shí)現(xiàn)equals() 和 hashCode()。業(yè)務(wù)鍵值相等的意思是,equals()方法 僅僅比較形成業(yè)務(wù)鍵的屬性,它能在現(xiàn)實(shí)世界里標(biāo)識(shí)我們的實(shí)例(是一個(gè)自然的候選碼)。
- public class Cat {
- ...
- public boolean equals(Object other) {
- if (this == other) return true;
- if ( !(other instanceof Cat) ) return false;
- final Cat cat = (Cat) other;
- if ( !cat.getLitterId().equals( getLitterId() ) ) return false;
- if ( !cat.getMother().equals( getMother() ) ) return false;
- return true;
- }
- public int hashCode() {
- int result;
- result = getMother().hashCode();
- result = 29 * result + getLitterId();
- return result;
- }
- }
注意:
業(yè)務(wù)鍵不必像數(shù)據(jù)庫(kù)的主鍵那樣固定不變。對(duì)業(yè)務(wù)鍵而言,不可變或唯一的屬性是不錯(cuò)的選擇。
【編輯推薦】