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
