@Data注解包含 -> @EqualsAndHashCode(callSuper = false)
@EqualsAndHashCode(callSuper = false) callsuper为false的含义是:在重写hashcode&equal时只重写当前类的数据(重点)
如果一个类上添加了@Data注解 并继承了某一个类,基于这个类创建的两个对象进行equal比较时, 会踩这个坑。
父类:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
private Integer id;
private String name;
}
子类:
@Data
public class User1 extends User {
private String username;
private Integer age;
User1(String username, Integer age, Integer id, String name) {
super(id, name);
this.username = username;
this.age = age;
}
}
测试类:
public class Test {
public static void main(String[] args) {
User1 user1 = new User1("yifan",18, 1, "俺是对象1");
User1 user2 = new User1("yifan",18, 2, "俺是对象2");
System.out.println(user1.hashCode());
System.out.println(user2.hashCode());
System.out.println(user1.equals(user2));
}
}
猜下user1和user2使用equal比较时 是true还是false?
来了!!!!!!
两个对象的hashcode相同 并且equal比较为true;
原因:@Data中包含@EqualsAndHashCode(callSuper = false) 它只会重写当前类的属性。
如果我们想让它即重写子类 又重写父类 该怎么办呢?
啊,问得好
@Data
@EqualsAndHashCode(callSuper = true)//覆盖了@Dataz中的@EqualsAndHashCode(callSuper = false)
public class User1 extends User {
private String username;
private Integer age;
User1(String username, Integer age, Integer id, String name) {
super(id, name);
this.username = username;
this.age = age;
}
}
覆盖后 这两个对象在用equal比较时就是false (hashcode也变了)
所以!!!!
慎重使用@Data 如果使用请添加上 @EqualsAndHashCode(callSuper = true)
或者说子类尽量不用@Data