该注解用于子类对象之间进行比较的时候
不加该注解的影响:子类对象属性值一致,但其继承的父类对象属性值不一致,在比较的时候会出现比较结果不对的情况。
举个简单的例子:
这边先定义一个分类对象 Parent,有一个属性:code
@Data
public class Parent {
/**
* 父类编码
*/
private String code;
}
再定义一个子类对象 Child,一一个属性:name
@Data
public class Child extends Parent {
/**
* 子类名称
*/
private String name;
}
在方法中 new 两个 Child 对象:childTest1、childTest2
对这两个 Child 对象的自有属性 name 都赋值为:Child;但是对继承的父类属性 code 进行不同的赋值
Child childTest1 = new Child();
childTest1.setCode("1");
childTest1.setName("child");
Child childTest2 = new Child();
childTest2.setCode("2");
childTest2.setName("child");
根据使用过程中,这两个对象肯定是不一样的,但是,在不加 @EqualsAndHashCode(callSuper = true) 注解的情况下对这两个对象进行比较得到的结果却是 true
boolean isSame = Objects.equals(childTest1,childTest2);
log.info("testEquals -> childTest1:{}, childTest2:{}", childTest1, childTest2);
log.info("testEquals -> :{}", isSame);
@EqualsAndHashCode(callSuper = true) 注解的作用就是将其父类属性也进行比较,下面是 Child 类加了注解后运行的结果:
@EqualsAndHashCode(callSuper = true)
@Data
public class Child extends Parent {
/**
* 子类名称
*/
private String name;
}
因为 @Data 生成的 equals 方法,只对该类里自有的属性进行了比较;
下面看下加与不加注解的时候编译后的 Child 类
(1)无 @EqualsAndHashCode(callSuper = true) 注解
public boolean equals(Object o){
if (o == this) {
return true;
}
if (!(o instanceof Child)) {
return false;
}
Child other = (Child)o;
if (!other.canEqual(this)) {
return false;
}
Object this$name = getName();Object other$name = other.getName();return this$name == null ? other$name == null : this$name.equals(other$name);
}
(2)有 @EqualsAndHashCode(callSuper = true) 注解
public boolean equals(Object o){
if (o == this) {
return true;
}
if (!(o instanceof Child)) {
return false;
}
Child other = (Child)o;
if (!other.canEqual(this)) {
return false;
}
if (!super.equals(o)) {
return false;
}
Object this$name = getName();Object other$name = other.getName();return this$name == null ? other$name == null : this$name.equals(other$name);
}
对比一下,可以看到加了注解之后多了 super.equals 方法
if (!super.equals(o)) {
return false;
}
细心的朋友会发现,在用 log 打印两个对象的时候,toString 方法只打印了子类属性,隐藏了父类属性,这里其实和 equals 方法一样,@Data 注解生成的 toString 方法也只包含了子类自有属性。
解决方案一样,加上 @ToString(callSuper = true) 注解,其实这里真正重要的是注解中的属性,callSuper = true,加上注解后打印结果如下:
以上