讀書筆記-《Effective Java》第9條:覆蓋equals時總要覆蓋hashCode
如果覆蓋equals方法卻不覆蓋hashCode方法,那麼就很有可能出現兩個物件equals方法返回true,但hashCode卻不一致的情況,例如:在HashMap中取不到正確的value。
HashMap的get方法是用hashCode匹配的。
public V get(Object key) {
Node<K,V> e;
return (e = getNode(hash(key), key)) == null ? null : e.value;
}
在重寫equals方法的時候重寫一下hashCode就可以了
public class People { public People() { } private String name = ""; private String age; @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj instanceof People) { People t = (People) obj; if (this.name == t.name) { return true; } } return false; } @Override public int hashCode() { int hash = 17; hash = hash * 31 + name.hashCode(); //關鍵 用equals判斷的屬性的hashCode拼 return hash; } }
用lombok方式
import lombok.EqualsAndHashCode;
@EqualsAndHashCode(exclude = {"age"})
public class People {
public People() {
}
private String name = "";
private String age;
}
lombok的@EqualsAndHashCode的說明,轉自:https://blog.csdn.net/zhanlanmg/article/details/50392266
原文中提到的大致有以下幾點:
1. 此註解會生成equals(Object other) 和 hashCode()方法。
2. 它預設使用非靜態,非瞬態的屬性
3. 可通過引數exclude排除一些屬性
4. 可通過引數of指定僅使用哪些屬性
5. 它預設僅使用該類中定義的屬性且不呼叫父類的方法
6. 可通過callSuper=true解決上一點問題。讓其生成的方法中呼叫父類的方法。
另:@Data相當於@Getter @Setter @RequiredArgsConstructor @ToString @EqualsAndHashCode這5個註解的合集。
通過官方文件,可以得知,當使用@Data註解時,則有了@EqualsAndHashCode註解,那麼就會在此類中存在equals(Object other) 和 hashCode()方法,且不會使用父類的屬性,這就導致了可能的問題。
比如,有多個類有相同的部分屬性,把它們定義到父類中,恰好id(資料庫主鍵)也在父類中,那麼就會存在部分物件在比較時,它們並不相等,卻因為lombok自動生成的equals(Object other) 和 hashCode()方法判定為相等,從而導致出錯
修復此問題的方法很簡單:
1. 使用@Getter @Setter @ToString代替@Data並且自定義equals(Object other) 和 hashCode()方法,比如有些類只需要判斷主鍵id是否相等即足矣。
2. 或者使用在使用@Data時同時加上@EqualsAndHashCode(callSuper=true)註解。