@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