当你在使用 Lombok 的 @Data 注解时,其实会有一些坑需要关注,今天就让我们来见识一下。
<!-- more -->

Lombok

先来简单介绍一下 Lombok ,其官方介绍如下:

Project Lombok makes java a spicier language by adding 'handlers' that know how to build and compile simple, boilerplate-free, not-quite-java code.

大致意思是 Lombok 通过增加一些"处理程序",可以让 Java 代码变得简洁、快速。

Lombok 提供了一系列的注解帮助我们简化代码,比如:

注解名称 功能
@Setter 自动添加类中所有属性相关的 set 方法
@Getter 自动添加类中所有属性相关的 get 方法
@Builder 使得该类可以通过 builder (建造者模式)构建对象
@RequiredArgsConstructor 生成一个该类的构造方法,禁止无参构造
@ToString 重写该类的toString()方法
@EqualsAndHashCode 重写该类的equals()hashCode()方法
@Data 等价于上面的@Setter@Getter@RequiredArgsConstructor@ToString@EqualsAndHashCode

看起来似乎这些注解都很正常,并且对我们的代码也有一定的优化,那为什么说@Data注解存在坑呢?

@Data注解

内部实现

由上面的表格我们可以知道,@Data是包含了@EqualsAndHashCode的功能,那么它究竟是如何重写equals()hashCode()方法的呢?

我们定义一个类TestA

@Data
public class TestA {String oldName;
}

我们将其编译后的 class 文件进行反编译:

public class TestA {String oldName;public TestA() {}public String getOldName() {return this.oldName;}public void setOldName(String oldName) {this.oldName = oldName;}public boolean equals(Object o) {// 判断是否是同一个对象if (o == this) {return true;}// 判断是否是同一个类else if (!(o instanceof TestA)) {return false;} else {TestA other = (TestA) o;if (!other.canEqual(this)) {return false;} else {// 比较类中的属性(注意这里,只比较了当前类中的属性)Object this$oldName = this.getOldName();Object other$oldName = other.getOldName();if (this$oldName == null) {if (other$oldName != null) {return false;}} else if (!this$oldName.equals(other$oldName)) {return false;}return true;}}}protected boolean canEqual(Object other) {return other instanceof TestA;}public int hashCode() {int PRIME = true;int result = 1;Object $oldName = this.getOldName();int result = result * 59 + ($oldName == null ? 43 : $oldName.hashCode());return result;}public String toString() {return "TestA(oldName=" + this.getOldName() + ")";}
}

针对其equals()方法,当它进行属性比较时,其实只比较了当前类中的属性。如果你不信的话,我们再来创建一个类TestB,它是TestA的子类:

@Data
public class TestB extends TestA {private String name;private int age;
}

我们将其编译后的 class 文件进行反编译:

public class TestB extends TestA {private String name;private int age;public TestB() {}public String getName() {return this.name;}public int getAge() {return this.age;}public void setName(String name) {this.name = name;}public void setAge(int age) {this.age = age;}public boolean equals(Object o) {if (o == this) {return true;} else if (!(o instanceof TestB)) {return false;} else {TestB other = (TestB)o;if (!other.canEqual(this)) {return false;} else {// 注意这里,真的是只比较了当前类中的属性,并没有比较父类中的属性Object this$name = this.getName();Object other$name = other.getName();if (this$name == null) {if (other$name == null) {return this.getAge() == other.getAge();}} else if (this$name.equals(other$name)) {return this.getAge() == other.getAge();}return false;}}}protected boolean canEqual(Object other) {return other instanceof TestB;}public int hashCode() {int PRIME = true;int result = 1;Object $name = this.getName();int result = result * 59 + ($name == null ? 43 : $name.hashCode());result = result * 59 + this.getAge();return result;}public String toString() {return "TestB(name=" + this.getName() + ", age=" + this.getAge() + ")";}
}

按照代码的理解,如果两个子类对象,其子类中的属性相同、父类中的属性不同时,利用equals()方法时,依旧会认为这两个对象相同,测试一下:

    public static void main(String[] args) {TestB t1 = new TestB();TestB t2 = new TestB();t1.setOldName("123");t2.setOldName("12345");String name = "1";t1.name = name;t2.name = name;int age = 1;t1.age = age;t2.age = age;System.out.println(t1.equals(t2));System.out.println(t2.equals(t1));System.out.println(t1.hashCode());System.out.println(t2.hashCode());System.out.println(t1 == t2);System.out.println(Objects.equals(t1, t2));}

结果为:

true
true
6373
6373
false
true

问题总结

对于父类是Object且使用了@EqualsAndHashCode(callSuper = true)注解的类,这个类由 Lombok 生成的equals()方法只有在两个对象是同一个对象时,才会返回 true ,否则总为 false ,无论它们的属性是否相同。

这个行为在大部分时间是不符合预期的,equals()失去了其意义。即使我们期望equals()是这样工作的,那么其余的属性比较代码便是累赘,会大幅度降低代码的分支覆盖率。

解决方法

  1. 用了@Data就不要有继承关系,类似 Kotlin 的做法。
  2. 自己重写equals(), Lombok 不会对显式重写的方法进行生成。
  3. 显式使用@EqualsAndHashCode(callSuper = true), Lombok 会以显式指定的为准。

Lombok中关于@Data的使用相关推荐

  1. lombok 中的@Data注解

    今天看到有代码中的Dao包中的类文件,写的极其简洁,甚至引起了开发工具InteliJ的报错,然后程序还能稳健地跑起来. 1 2 3 4 5 6 7 8 9 import lombok.Data; @D ...

  2. lombok中的@Data注解与MyBatis的懒加载机制冲突解决

    使用@Data注解与mybatis的懒加载机制实现一对一关系查询时,发现怎么配置都无效,就是一下都查出来了,根本没有懒加载 1.application.yml配置文件配置如下: # mybatis 配 ...

  3. 18.案例实战:体验lombok的核心@Data和@Slf4j注解

    代码:https://github.com/NIGHTFIGHTING/spring_boot_learning/tree/master/18/agan-boot/agan-boot-lombok 添 ...

  4. java socket中属性详解_前端开发:关于Vue组件中的data属性值是函数而不是对象的详解...

    最近在搞关于前端开发的基础知识归纳,发现了不少经典知识点,那么本篇博文就来分享一个经典的知识点:Vue组件中的data属性值为什么是函数而不是对象.首先来了解一下Vue组件的使用理念:在Vue组件使用 ...

  5. ECharts问题--柱状图和折线图中xAxis.data为空时报错问题解决

    ECharts问题--柱状图和折线图中xAxis.data为空时报错问题解决 参考文章: (1)ECharts问题--柱状图和折线图中xAxis.data为空时报错问题解决 (2)https://ww ...

  6. 解决IDEA使用lombok注解无效,@Data不生效问题

    解决IDEA使用lombok注解无效,@Data不生效问题 参考文章: (1)解决IDEA使用lombok注解无效,@Data不生效问题 (2)https://www.cnblogs.com/houz ...

  7. WinCE中的Data/ Prefetch Abort异常定位

    WinCE中的Data/ Prefetch Abort等异常定位 在调试WinCE程序的时候,有时候会碰到Data/Prefetch Abort的异常,相信从事过WinCE开发的人对这种异常信息应该都 ...

  8. eclipse工具连接mysql_eclipse工具中使用Data Source Explorer连接数据库(MySQL)

    1.进入Eclipse工具,打开Data Source Explorer.Window==>Show View==>Data Source Explorer(注:如果找不到请选择Other ...

  9. lombok插件:Data自动get/set方法, Slf4j实现Logger的调用

    lombok插件:Data自动get/set方法, Slf4j实现Logger的调用 lombok.Data import lombok.Data; import org.hibernate.anno ...

最新文章

  1. ACMNO.49:一元三次方程求解(主要就是精度问题)
  2. 各类木材强度_层状磷酸锆/ 聚磷酸铵复合阻燃剂对木材的阻燃抑烟性能研究
  3. Database之SQLSever:SQL命令实现四则运算、desc降序、like模糊查询、distinct去重、MAX/MIN/SUM/AVG/COUNT/GROUP/having等案例之详细攻略
  4. 区块链BaaS云服务(12)易居(中国) 房地产 EBaaS(Estate Blockchain as a Service)
  5. 04-JDBC连接MySQL数据库【修改数据】
  6. 数字n,按字典排序,找出第k小的数字
  7. Python_48re模块的sub方法
  8. exec和source的区别
  9. Visual Studio 2017正式发布
  10. 计算机硬盘容量的最小单位,计算机中存储数据的最小单位和存储容量的基本单位各是什么?...
  11. 中卫市地图arcgis数据shp道路地名县区边界水系2021年(下载说明)
  12. 图片放大后不清晰怎么处理?
  13. 储存卡数据怎么恢复?教你几招解决
  14. python 协程 asyncio_Python 原生协程------asyncio(选自公众号)
  15. 电脑开机时stage 1 fo 3 stage 2 fo 3 Checking file system on D:
  16. C++[缺省参数]的理解
  17. 一台服务器,启动多个redis
  18. 大数据必学Java基础(六十一):同步类容器对比应用
  19. R在市场调查中的应用--主成分分析
  20. 【eclipse】版本代号

热门文章

  1. Django Cookie于Session
  2. 上白泽慧音(tarjan,图的染色)
  3. 1.1图像处理的概念
  4. 万水千山ABP - 弹出对话框禁用回车
  5. 1021. 个位数统计 (15)
  6. 从svn导入项目后处理一些报错信息
  7. 工作中常用到的一些方法集合
  8. CG-CTF-Web-AAencode
  9. Git使用教程:真正手把手教你使用git!
  10. Vue 中 slot插槽 的使用