美文网首页Java架构技术进阶Java
SpringBoot初始化几大招式,看了终于明白了

SpringBoot初始化几大招式,看了终于明白了

作者: Java入门到入坟 | 来源:发表于2020-02-26 16:15 被阅读0次
推荐阅读:阿里二面凉经:虚拟机+MySQL+中间件+设计模式+缓存+Spring+并发等难题,全部迎刃而解

背景

在日常开发时,我们常常需要 在SpringBoot 应用启动时执行某一段逻辑,如下面的场景:

  • 获取一些当前环境的配置或变量

  • 向数据库写入一些初始数据

  • 连接某些第三方系统,确认对方可以工作..

在实现这些功能时,我们可能会遇到一些"坑"。为了利用SpringBoot框架的便利性,我们不得不将整个应用的执行控制权交给容器,于是造成了大家对于细节是一无所知的。那么在实现初始化逻辑代码时就需要小心了,比如,我们并不能简单的将初始化逻辑在Bean类的构造方法中实现,类似下面的代码:

@Component
public  class  InvalidInitExampleBean {

     @Autowired
     private  Environment env;

     public  InvalidInitExampleBean() {
          env.getActiveProfiles();
     }
}

这里,我们在InvalidInitExampleBean的构造方法中试图访问一个自动注入的env字段,当真正执行时,你一定会得到一个空指针异常(NullPointerException)。

原因在于,当构造方法被调用时,Spring上下文中的Environment这个Bean很可能还没有被实例化,同时也仍未注入到当前对象,所以并不能这样进行调用。

下面,我们来看看在SpringBoot中实现"安全初始化"的一些方法:

1、 @PostConstruct 注解

@PostConstruct 注解其实是来自于 javax的扩展包中(大多数人的印象中是来自于Spring框架),它的作用在于声明一个Bean对象初始化完成后执行的方法

来看看它的原始定义:

The PostConstruct annotation is used on a method that needs to be executed after dependency injection is done to perform any initialization

也就是说,该方法会在所有依赖字段注入后才执行,当然这一动作也是由Spring框架执行的。

下面的代码演示了使用@PostConstruct的例子:

2、 InitializingBean 接口

InitializingBean 是由Spring框架提供的接口,其与@PostConstruct注解的工作原理非常类似。如果不使用注解的话,你需要让Bean实例继承 InitializingBean接口,并实现afterPropertiesSet()这个方法。

下面的代码,展示了这种用法:

3、 @Bean initMethod方法

我们在声明一个Bean的时候,可以同时指定一个initMethod属性,该属性会指向Bean的一个方法,表示在初始化后执行。

如下所示:

然后,这里将initMethod指向init方法,相应的我们也需要在Bean中实现这个方法:

上面的代码是基于Java注解的方式,使用Xml配置也可以达到同样的效果:

该方式在早期的 Spring版本中大量被使用

4、 构造器注入

如果依赖的字段在Bean的构造方法中声明,那么Spring框架会先实例这些字段对应的Bean,再调用当前的构造方法。此时,构造方法中的一些操作也是安全的,如下:

5、 ApplicationListener

ApplicationListener 是由 spring-context组件提供的一个接口,主要是用来监听 "容器上下文的生命周期事件"。它的定义如下:

这里的event可以是任何一个继承于ApplicationEvent的事件对象。对于初始化工作来说,我们可以通过监听ContextRefreshedEvent这个事件来捕捉上下文初始化的时机。如下面的代码:

在Spring上下文初始化完成后,这里定义的方法将会被执行。与前面的InitializingBean不同的是,通过ApplicationListener监听的方式是全局性的,也就是当所有的Bean都初始化完成后才会执行方法。

Spring 4.2 之后引入了新的 @EventListener注解,可以实现同样的效果:

6、 CommandLineRunner

SpringBoot 提供了一个CommanLineRunner接口,用来实现在应用启动后的逻辑控制,其定义如下:

这里的run方法会在Spring 上下文初始化完成后执行,同时会传入应用的启动参数。如下面的代码:

此外,对于多个CommandLineRunner的情况下可以使用@Order注解来控制它们的顺序。

7、 ApplicationRunner

与 CommandLineRunner接口类似, Spring boot 还提供另一个ApplicationRunner 接口来实现初始化逻辑。不同的地方在于 ApplicationRunner.run()方法接受的是封装好的ApplicationArguments参数对象,而不是简单的字符串参数。

ApplicationArguments对象提供了一些非常方便的方法,可以用来直接获取解析后的参数,比如:

java -jar application.jar --debug --ip=xxxx

此时通过 ApplicationArguments的getOptionNames就会得到["debug","ip"]这样的值。

测试代码

下面,通过一个小测试来演示几种初始化方法的执行次序。

按如下代码实现一个复合式的Bean:

执行这个Bean的初始化,会发现日志输出如下:

所以,这几种初始化的顺序为:

  1. 构造器方法

  2. @PostConstruct 注解方法

  3. InitializingBean的afterPropertiesSet()

  4. Bean定义的initMethod属性方法

来源:https://mp.weixin.qq.com/s/XJ8wGiN9LTYA_i7Z1YB3CQ

相关文章

  • SpringBoot初始化几大招式,看了终于明白了

    推荐阅读:阿里二面凉经:虚拟机+MySQL+中间件+设计模式+缓存+Spring+并发等难题,全部迎刃而解 背景 ...

  • SpringBoot启动源码解读

    最近看springBoot,研究一下springBoot启动源码 springboot启动 初始化SpringAp...

  • 终于明白了

    小时​候​一直不​理‎解 ‎爸妈为什​么‎那‎么早起‎​床 最难的时候三更起床,半夜从地里回家 那时候…… ​长‎...

  • 终于明白了

    闭上双眼、静心聆听: 在一个漆黑伸手不见五指的夜晚,一条狭窄的小巷中,一个路人被人撞上之后,突然看...

  • 终于明白了

    下午,我被安排了一项“本职工作”,就是可以完全把我的专业给发挥出来的活儿。于是,我很认真并且花费了两个小时把它完成...

  • 终于明白了

    主课论文一直在边缘环节徘徊的原因,我终于想明白了… 这其中最大的阻力就是不想开始做形态分析,这大概是被本科毕业论文...

  • 终于明白了

    《圣经》记载:“其实明天如何,你们还不知道。你们的生命是什么呢?你们原来是一片云雾,出现少时就不见了。 ”(雅各书...

  • 终于明白了

    你们知道我小时候最害怕的事是什么吗? 我小时候,我的爸爸也是年轻力壮得中年男人,妈妈是顾家的合格妻子和母...

  • 慕课网高并发实战(三)- 项目准备

    课程网址 项目采用springboot基础框架快速开发,具体初始化方法不再赘述,不明白的可以看课程或者自行百度 1...

  • 看了爽文改编的剧以后终于明白了……

    好羡慕啊 不用奋斗的人生 躺平 躺赢 终于理解爽文为什么那么赢得人心了 毕竟大部分人生活在底层 上流的影子都看不到...

网友评论

    本文标题:SpringBoot初始化几大招式,看了终于明白了

    本文链接:https://www.haomeiwen.com/subject/egpcqhtx.html