加载顺序_Spring bean 加载顺序导致的 bug 问题

本文详细介绍了在 Spring Boot 项目中遇到的配置文件属性加载问题,由于 Bean 的加载顺序导致 @Value 注解获取的属性值为 null。分析了 @Configuration、@Service 和 @Component 的加载顺序,并指出默认加载顺序基于包名+文件名。通过 @DependsOn 注解可以设定 Bean 的加载顺序,以确保依赖的 Bean 在需要时已经初始化。此外,还提及了 @PostConstruct 注解在初始化过程中的作用。
摘要由CSDN通过智能技术生成

9c3f11718871a5b7af71e72ffde7ee86.png

一、问题描述

今天启动 spring boot 项目的时候,有时候会报加载不到配置文件的属性。配置文件的属性是用 @Value 获取的,属性有时候会是 null 。

程序经过简化,是这样的,有一个 InitConfig 类,用来让静态工具类能获取到配置文件的属性值。内容是这样的:

7f7a2685162b4c44dfaef8f32e6cb3a8.png

在静态工具类中,通过 InitConfig.load(); 来获取配置文件中的属性值,这是没问题的,因为 @Configuration 类会在 spring 程序启动过程中就执行了。

但如果在 @Service 修饰的类中,调用 InitConfig.load(); 如下图所示:

c14e97929a6ad69fa32f83ac55b49b9e.png

这样,有时候就会获取不到配置文件中的属性值。如下图所示:

5c837bdce3f53497f12a5b45188f2089.png

很奇怪,经过研究尝试,终于了解了其中的缘由。现在给大家分享一下。

二、spring bean 加载顺序

之前我一直以为 @Configuration 会比 @Service、@Component 优先执行。其实不对。看下面的代码片段:

文件结构:

505409f39ad90ddf1c197f532e701712.png

Aaa.java 文件:

77fbcca948ba8860af1efb05b2bab331.png

Bb.java 文件:

13c89f83d897f7291f835f34b7f281d4.png

再结合上面的 InitConfig.java 文件。当项目启动的过程中,你会发现这样的结果:

72411763212a44e0a98edab19aecfa52.png

Aaa.java 先执行,Bb.java 其次,InitConfig.java 文件最后执行。这样就验证了 @Configuration 并不会比 @Service、@Component 优先执行。

我猜测的应该是,spring 将上面带有注解的类都放在一起,统一加载。默认是根据 包名+文件名称 来判断加载顺序的。

@Configuration、@Service、@Component 都会将修饰的类交给 spring 来管理,文件初始化的时候,会加载属性,无参构造方法等。

三、设置 spring bean 加载顺序

有这么一个注解,@DependsOn,它可以指定依赖哪个 bean ,让自己在该 bean 之后加载。这样就可以实现 bean 顺序的设置。

@Configuration
@DependsOn({"initConfig", "aaa"})
public class Bb {
    ...
}

@DependsOn 可以指定多个 bean ,用 String[] 表示,有顺序。@DependsOn({"initConfig", "aaa"}) 表示在执行 Bb.java 之前,会首先执行 InitConfig.java,然后再执行 Aaa.java。bean 名称默认为 首字母小写的文件名。

四、小结

@Configuration、@Service、@Component 都会将修饰的类交给 spring 来管理,但就注解这个层面来说,貌似是没有加载顺序的。默认为 包名+文件名 来判断加载顺序。

如果需要指定加载顺序,可以使用 @DependsOn 注解。

文中还用到了 @PostConstruct 注解。它是 jdk 中的一个注解, 被 @PostConstruct 修饰的方法会在服务器加载 Servlet 的时候运行,并且只会被服务器调用一次。


感谢大家看到最后,我是 不务正业的程序汪,文章每天持续更新!欢迎大家指出我的文章的不足之处,也欢迎大家关注、点赞+转发。
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值