1. 程式人生 > >讀書筆記-《Effective Java》第9條:覆蓋equals時總要覆蓋hashCode

讀書筆記-《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)註解。