Lombok中@Data的坑

探讨了在SpringBoot升级过程中遇到的Lombok@Data注解引发的编译错误,详细分析了从2.0.2到2.0.3版本Lombok版本变更导致的类加载顺序问题,并提供了解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

    最近有同事在spring boot中用Lombok @Data注解时遇到了一个奇怪的问题,然后有幸一起研究了一下,把研究成果记录下来。

问题

先上代码:

@Data
public abstract class TestAbstract {

    private RedisTemplate redisTemplate;

    public TestAbstract(RedisTemplate redisTemplate) {
        this.redisTemplate = redisTemplate;
    }

}
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
@Data
public class TestChild extends TestAbstract {

    @Autowired
    public TestChild(RedisTemplate redisTemplate) {
        super(redisTemplate);
    }
}
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

上面的代码在spring boot 2.0.2版本可以正常运行,但在2.0.3版本却编译错误。

Error:(13, 1) java: TestAbstract() 在 com.brotherj.learn.TestAbstract 中是private 访问控制
 
  • 1

查看了pom.xml文件后发现lombok的版本号是从spring boot继承而来:

<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <optional>true</optional>
</dependency>
 
  • 1
  • 2
  • 3
  • 4
  • 5

然后又查看了spring boot pom.xml中的lombok版本号,发现从spring boot 2.0.3版本开始使用了lombok 1.16.22,而2.0.2版本使用的是1.16.20版本。

#2.0.2
<lombok.version>1.16.20</lombok.version>

#2.0.3
<lombok.version>1.16.22</lombok.version>
 
  • 1
  • 2
  • 3
  • 4
  • 5

于是修改pom.xml版本号之后,问题解决。

 <dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <optional>true</optional>
    <version>1.16.20</version>
</dependency>
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

原因

问题解决后,决定研究一下具体的原因,由于Lombok只是依靠可插件化的Java自定义注解处理API(JSR 269: Pluggable Annotation Processing API)来实现在Javac编译阶段利用“Annotation Processor”对自定义的注解进行预处理后生成真正在JVM上面执行的“Class文件”。所以应该时两个版本下编译生成的class文件不同,于是比较了一下两个版本下编译后的class文件。

    public TestAbstract(RedisTemplate redisTemplate) {
        this.redisTemplate = redisTemplate;
    }
    //1.16.22会生成一个private的构造方法,而1.16.20不会生成
    private TestAbstract() {
    }

     @Autowired
    public TestChild(RedisTemplate redisTemplate) {
        super(redisTemplate);
    }

    private TestChild() {
    }

 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

看到这里一切就都清楚了,是类加载顺序导致的问题。先复习一下类加载顺序:(静态变量、静态初始化块)–>(变量、初始化块)–> 构造器。如果有父类,加载顺序是:父类static方法 –> 子类static方法 –> 父类构造方法- -> 子类构造方法 。也就是说,当加载TestChild类时先加载类TestAbstract的构造方法,而类TestAbstract的空参构造为private,于是报了访问控制的错误。

原文地址 : https://blog.youkuaiyun.com/sinat_29899265/article/details/82493944

### Lombok 注解 使用问题及解决方案 #### 版本兼容性问题 当遇到Lombok注解失效的情况时,可能是由于使用的Lombok版本较低所致。将Lombok版本升级至1.18.24可以有效解决问题[^1]。 #### 开发环境配置不当 在某些开发环境中,即使已经添加了必要的依赖项和注解,仍可能出现Lombok功能未正常工作的情形。对于这种情况,确认IDEA中已正确安装Lombok插件是非常重要的一步。如果是在较高版本的IDEA(如2020.1及以上)中未能找到该插件,则可能是因为网络连接不稳定造成的;此时可以选择离线方式下载并手动完成插件的安装过程[^5]。 #### 构建工具设置缺失 针对采用Gradle作为构建工具的项目而言,仅仅引入Lombok库并不足以确保其顺利运作。还需要特别注意的是,在`build.gradle`文件里加入相应的注解处理器配置,这样才能让编译器识别到这些特殊的Java语法糖,并据此自动生成所需的getter/setter等辅助函数[^4]。 ```groovy dependencies { compileOnly 'org.projectlombok:lombok:1.18.24' } annotationProcessor 'org.projectlombok:lombok:1.18.24' ``` #### 验证注解应用情况 为了进一步排查具体哪一部分出现了错误,可以在项目的`target`目录下寻找目标类文件,检查其中是否确实包含了由Lombok生成的方法定义。假如发现即便有相应方法也依旧无法调用成功的话,那么应当仔细核对源码中的注解声明部分——比如确认是否遗漏了像`@Getter`, `@Setter`或是更便捷的一站式服务型注解`@Data`这样的关键字[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值