
点击上方蓝色“大数据实战演练”,选择“设为星标”或“置顶”java
回复“资源”领取独家整理的学习资料!web
每个成功人士的背后,一定曾经作出过勇敢而又孤独的决定。spring
放弃不难,但坚持很酷~sql
1、问题描述
今天启动 spring boot 项目的时候,有时候会报加载不到配置文件的属性。配置文件的属性是用 @Value 获取的,属性有时候会是 null 。npm
程序通过简化,是这样的,有一个 InitConfig 类,用来让静态工具类能获取到配置文件的属性值。内容是这样的:c#

在静态工具类中,经过 InitConfig.load(); 来获取配置文件中的属性值,这是没问题的,由于 @Configuration 类会在 spring 程序启动过程当中就执行了。后端
但若是在 @Service 修饰的类中,调用 InitConfig.load(); 以下图所示:服务器

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

很奇怪,通过研究尝试,终于了解了其中的原因。如今给你们分享一下。app
2、spring bean 加载顺序
以前我一直觉得 @Configuration 会比 @Service、@Component 优先执行。其实不对。看下面的代码片断:
文件结构:

Aaa.java 文件:

Bb.java 文件:

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

Aaa.java 先执行,Bb.java 其次,InitConfig.java 文件最后执行。这样就验证了 @Configuration 并不会比 @Service、@Component 优先执行。
我猜想的应该是,spring 将上面带有注解的类都放在一块儿,统一加载。默认是根据 包名+文件名称 来判断加载顺序的。
@Configuration、@Service、@Component 都会将修饰的类交给 spring 来管理,文件初始化的时候,会加载属性,无参构造方法等。
3、设置 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 名称默认为 首字母小写的文件名。
4、小结
@Configuration、@Service、@Component 都会将修饰的类交给 spring 来管理,但就注解这个层面来讲,貌似是没有加载顺序的。默认为 包名+文件名 来判断加载顺序。
若是须要指定加载顺序,可使用 @DependsOn 注解。
文中还用到了 @PostConstruct 注解。它是 jdk 中的一个注解, 被 @PostConstruct 修饰的方法会在服务器加载 Servlet 的时候运行,而且只会被服务器调用一次。
好啦,以上基本就是对 Spring bean 加载顺序致使问题 bug 的思考,若是上述描述有欠缺或错误,欢迎指正,感谢。
往期推荐
扫一扫,咱们的故事就开始了。
若是这篇文章对你有所启发,点赞、转发都是一种支持!
让我知道你在看

本文分享自微信公众号 - 大数据实战演练(gh_f942bfc92d26)。
若有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一块儿分享。