@EqualsAndHashCode


1、@Data注解包含了這些注解

* @see Getter
* @see Setter
* @see RequiredArgsConstructor
* @see ToString
* @see EqualsAndHashCode
* @see lombok.Value

1. 此注解會生成equals(Object other) 和 hashCode()方法。 
2. 它默認使用非靜態,非瞬態的屬性 
3. 可通過參數exclude排除一些屬性 
4. 可通過參數of指定僅使用哪些屬性 
5. 它默認僅使用該類中定義的屬性且不調用父類的方法 
6. 可通過callSuper=true解決上一點問題。讓其生成的方法中調用父類的方法。

另:@Data相當於@Getter @Setter @RequiredArgsConstructor @ToString @EqualsAndHashCode這5個注解的合集。

 

2、@Data就包含了EqualsAndHashCode 重寫了hashcode和equals方法

3、EqualsAndHashCode  對於繼承父類屬性時需要注意的點

通過官方文檔,可以得知,當使用@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)注解。

 

4、測試

@Data
//@EqualsAndHashCode(callSuper=true)
public class UcsAccountLog extends  BaseVO implements Serializable {
 
    private static final long serialVersionUID = 1L;
 
    private String id;
    
}

 

@Data
public class BaseVO {
    public String version;
}

 

/**
     * 首先,是為了支持哈希表類的如之類的底層使用了哈希表的類。
     * Java Object規范中int hashCode()方法的約定類容有三個:
     * <p>
     * (1)              只要對象的equals方法所用到的信息沒有修改,那么hashCode方法必須始終如一的返回一個同一整數,在同一應用程序中多次執行中,每一次執行可以不一樣。
     * <p>
     * (2)              如果兩個對象的equals方法想到,那么每一個對象單獨調用hashCode返回的數字必須相等。
     * <p>
     * (3)              如果兩個對象的equals方法不相等,hashCode方法的返回值可以相等,給不同的對象返回不同的值可以提高哈希表的性能。
     * 所有重寫 equals的同時 必須重寫hashcode
     * @param args
     */
    public static void main(String[] args) {
        UcsAccountLog ucsAccountLog = new UcsAccountLog();
        ucsAccountLog.setVersion("1");
        ucsAccountLog.setId("2");
        UcsAccountLog ucsAccountLog2 = new UcsAccountLog();
        ucsAccountLog2.setVersion("2");
        ucsAccountLog2.setId("2");
        HashSet set = new HashSet();
        set.add(ucsAccountLog);
        set.add(ucsAccountLog2);
        System.out.println(set.size());//1
        System.out.println(ucsAccountLog.hashCode() == ucsAccountLog2.hashCode());//true  因為重寫了hashcode 所有這里hashcode是相等的
        //如果沒有重寫 hashcode是根據內存地址值
        System.out.println(ucsAccountLog.equals(ucsAccountLog2));//true
        //根據hash
        System.out.println(ucsAccountLog == ucsAccountLog2);//false  ==比較的是對象的地址
 
    }

 如果子類 

//@EqualsAndHashCode(callSuper=true)

沒有添加這一行,則輸出結果是 true true false 和我們期望的結果不符合 

 因為@EqualsAndHashCode默認不繼承父類 就是說 重寫hashcode和equals的時候 不包含父類的字段值,所有只會比較自己對象里面的字段值  顯然這是錯誤的

打開注解 則輸出正確結果 false false false

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM