Lombok中关于@Data的使用
当你在使用 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()
是这样工作的,那么其余的属性比较代码便是累赘,会大幅度降低代码的分支覆盖率。
解决方法
- 用了
@Data
就不要有继承关系,类似 Kotlin 的做法。 - 自己重写
equals()
, Lombok 不会对显式重写的方法进行生成。 - 显式使用
@EqualsAndHashCode(callSuper = true)
, Lombok 会以显式指定的为准。
Lombok中关于@Data的使用相关推荐
- lombok 中的@Data注解
今天看到有代码中的Dao包中的类文件,写的极其简洁,甚至引起了开发工具InteliJ的报错,然后程序还能稳健地跑起来. 1 2 3 4 5 6 7 8 9 import lombok.Data; @D ...
- lombok中的@Data注解与MyBatis的懒加载机制冲突解决
使用@Data注解与mybatis的懒加载机制实现一对一关系查询时,发现怎么配置都无效,就是一下都查出来了,根本没有懒加载 1.application.yml配置文件配置如下: # mybatis 配 ...
- 18.案例实战:体验lombok的核心@Data和@Slf4j注解
代码:https://github.com/NIGHTFIGHTING/spring_boot_learning/tree/master/18/agan-boot/agan-boot-lombok 添 ...
- java socket中属性详解_前端开发:关于Vue组件中的data属性值是函数而不是对象的详解...
最近在搞关于前端开发的基础知识归纳,发现了不少经典知识点,那么本篇博文就来分享一个经典的知识点:Vue组件中的data属性值为什么是函数而不是对象.首先来了解一下Vue组件的使用理念:在Vue组件使用 ...
- ECharts问题--柱状图和折线图中xAxis.data为空时报错问题解决
ECharts问题--柱状图和折线图中xAxis.data为空时报错问题解决 参考文章: (1)ECharts问题--柱状图和折线图中xAxis.data为空时报错问题解决 (2)https://ww ...
- 解决IDEA使用lombok注解无效,@Data不生效问题
解决IDEA使用lombok注解无效,@Data不生效问题 参考文章: (1)解决IDEA使用lombok注解无效,@Data不生效问题 (2)https://www.cnblogs.com/houz ...
- WinCE中的Data/ Prefetch Abort异常定位
WinCE中的Data/ Prefetch Abort等异常定位 在调试WinCE程序的时候,有时候会碰到Data/Prefetch Abort的异常,相信从事过WinCE开发的人对这种异常信息应该都 ...
- eclipse工具连接mysql_eclipse工具中使用Data Source Explorer连接数据库(MySQL)
1.进入Eclipse工具,打开Data Source Explorer.Window==>Show View==>Data Source Explorer(注:如果找不到请选择Other ...
- lombok插件:Data自动get/set方法, Slf4j实现Logger的调用
lombok插件:Data自动get/set方法, Slf4j实现Logger的调用 lombok.Data import lombok.Data; import org.hibernate.anno ...
最新文章
- ACMNO.49:一元三次方程求解(主要就是精度问题)
- 各类木材强度_层状磷酸锆/ 聚磷酸铵复合阻燃剂对木材的阻燃抑烟性能研究
- Database之SQLSever:SQL命令实现四则运算、desc降序、like模糊查询、distinct去重、MAX/MIN/SUM/AVG/COUNT/GROUP/having等案例之详细攻略
- 区块链BaaS云服务(12)易居(中国) 房地产 EBaaS(Estate Blockchain as a Service)
- 04-JDBC连接MySQL数据库【修改数据】
- 数字n,按字典排序,找出第k小的数字
- Python_48re模块的sub方法
- exec和source的区别
- Visual Studio 2017正式发布
- 计算机硬盘容量的最小单位,计算机中存储数据的最小单位和存储容量的基本单位各是什么?...
- 中卫市地图arcgis数据shp道路地名县区边界水系2021年(下载说明)
- 图片放大后不清晰怎么处理?
- 储存卡数据怎么恢复?教你几招解决
- python 协程 asyncio_Python 原生协程------asyncio(选自公众号)
- 电脑开机时stage 1 fo 3 stage 2 fo 3 Checking file system on D:
- C++[缺省参数]的理解
- 一台服务器,启动多个redis
- 大数据必学Java基础(六十一):同步类容器对比应用
- R在市场调查中的应用--主成分分析
- 【eclipse】版本代号