Spring多种启动初始化方案,看这篇就够了

开发 架构
我们经常有一些业务需求,需要在项目启动后执行相关的业务代码,如:数据的初始化业务。今天我们来梳理一下有哪些方案?

[[410656]]

目录

  • 前言
  • CommandLineRunner
  • ApplicationRunner
  • ApplicationListener
  • @PostConstruct
  • InitializationBean
  • 总结

前言

我们经常有一些业务需求,需要在项目启动后执行相关的业务代码,如:数据的初始化业务。今天我们来梳理一下有哪些方案?

CommandLineRunner

CommandLineRunner是一个接口,通过实现它,我们可以在Spring应用成功启动之后执行一些代码片段

我们先定义个User实体Bean

 

Spring多种启动初始化方案,看这篇就够了

 

下面我们定义一个类实现CommandLineRunner接口

 

Spring多种启动初始化方案,看这篇就够了

 

当 Spring Boot 在应用上下文中找到 CommandLineRunner bean,它将会在应用成功启动之后调用 run() 方法,并传递用于启动应用程序的命令行参数

java -jar demo-0.0.1-SNAPSHOT.jar --foo=bar --name=gujch

启动执行结果

 

Spring多种启动初始化方案,看这篇就够了

 

小结:

  1. 命令行传入的参数并没有被解析,而只是显示出我们传入的字符串内容 --foo=bar,--name=gujch
  2. 在重写的 run() 方法上有 throws Exception 标记,Spring Boot 会将 CommandLineRunner 作为应用启动的一部分,如果运行 run() 方法时抛出 Exception,应用将会终止启动
  3. 当有多个 CommandLineRunner 时,将会按照 @Order 注解中的数字从小到大

如果我们只是想简单的获取以空格分隔的命令行参数,那 MyCommandLineRunner 就足够使用了

ApplicationRunner

上面提到,通过命令行启动并传递参数,MyCommandLineRunner 不能解析参数,如果要解析参数,那我们就要用到 ApplicationRunner 参数了

 

Spring多种启动初始化方案,看这篇就够了

 

执行结果

 

Spring多种启动初始化方案,看这篇就够了

 

到这里我们可以看出:

同 MyCommandLineRunner 相似,但 ApplicationRunner 可以通过 run 方法的 ApplicationArguments 对象解析出命令行参数,并且每个参数可以有多个值在里面,因为 getOptionValues 方法返回 List数组

在重写的 run() 方法上有 throws Exception 标记,Spring Boot 会将 ApplicationRunner 作为应用启动的一部分,如果运行 run() 方法时抛出 Exception,应用将会终止启动

ApplicationRunner 也可以使用 @Order 注解进行排序,从启动结果来看,它与 CommandLineRunner 共享 order 的顺序

我们来看看源码,CommandLineRunner 和 ApplicationRunner 是在何时被调用的呢?

SpringApplication.java类中callRunners方法

 

Spring多种启动初始化方案,看这篇就够了

 

上面可以看到spring获取CommandLineRunner 和 ApplicationRunner Bean会放到List中,然后一起排序,所以@Order排序是共享的

ApplicationListener

如果我们不需要获取命令行参数时,我们可以将启动逻辑绑定到 Spring 的 ApplicationReadyEvent 上

 

Spring多种启动初始化方案,看这篇就够了

 

执行结果

 

Spring多种启动初始化方案,看这篇就够了

 

ApplicationReadyEvent 当且仅当 在应用程序就绪之后才被触发。

启动顺序Order不与CommandLineRunner和ApplicationRunner共享

如果我们不需要获取命令行参数,我们可以通过 ApplicationListener 创建一些全局的启动逻辑,我们还可以通过它获取 Spring Boot 支持的 configuration properties 环境变量参数 ,因为event参数有configuration上下文

 

Spring多种启动初始化方案,看这篇就够了

 

@PostConstruct

创建启动逻辑的另一种简单解决方案是提供一种在 bean 创建期间由 Spring 调用的初始化方法。我们要做的就只是将 @PostConstruct 注解添加到方法中:

 

Spring多种启动初始化方案,看这篇就够了

 

执行结果

 

Spring多种启动初始化方案,看这篇就够了

 

从上面运行结果可以看出:

1)Spring 创建完 bean之后 (在启动之前),便会立即调用 @PostConstruct 注解标记的方法,因此我们无法使用 @Order 注解对其进行自由排序,因为它可能依赖于 @Autowired插入到我们 bean 中的其他 Spring bean。

2)相反,它将在依赖于它的所有 bean 被初始化之后被调用

@PostConstruct 方法固有地绑定到现有的 Spring bean,因此应仅将其用于此单个 bean 的初始化逻辑;

@PostConstruct应用场景:

在生成对象时候做一些初始化操作,而这些初始化操作又依赖于依赖注入(populateBean),那么就无法在构造函数中实现。这时,可以使用@PostConstruct注解一个方法来完成初始化,@PostConstruct注解的方法将会在依赖注入完成后被自动调用。

InitializingBean

与 @PostConstruct 解决方案非常相似,我们可以实现 InitializingBean 接口,并让 Spring 调用某个初始化方法:

 

Spring多种启动初始化方案,看这篇就够了

 

执行结果

 

Spring多种启动初始化方案,看这篇就够了

 

@PostConstruct 和 afterPropertiesSet 区别

1、afterPropertiesSet,顾名思义「在属性设置之后」,调用该方法时,该 bean 的所有属性已经被 Spring 填充。如果我们在某些属性上使用 @Autowired(常规操作应该使用构造函数注入),那么 Spring 将在调用afterPropertiesSet 之前将 bean 注入这些属性。但 @PostConstruct 并没有这些属性填充限制

2、所以

InitializingBean.afterPropertiesSet 解决方案比使用 @PostConstruct 更安全,因为如果我们依赖尚未自动注入的 @Autowired 字段,则 @PostConstruct 方法可能会遇到 NullPointerExceptions

总结

从上面的例子中我们就可以发现各个启动方案的顺序

针对Bean实体启动初始化 顺序

Construct >> @Autowired(依赖注入) >> @postConstruct >> InitializingBean

针对整体项目启动 顺序

 

CommandLineRunner和ApplicationRunner >> ApplicationListener

 

责任编辑:武晓燕 来源: 今日头条
相关推荐

2024-08-27 11:00:56

单例池缓存bean

2017-03-30 22:41:55

虚拟化操作系统软件

2021-09-30 07:59:06

zookeeper一致性算法CAP

2019-08-16 09:41:56

UDP协议TCP

2020-09-18 16:37:59

数据可视化技术Python

2022-03-29 08:23:56

项目数据SIEM

2021-05-07 07:52:51

Java并发编程

2023-09-25 08:32:03

Redis数据结构

2023-10-04 00:32:01

数据结构Redis

2021-09-10 13:06:45

HDFS底层Hadoop

2023-11-07 07:46:02

GatewayKubernetes

2021-07-28 13:29:57

大数据PandasCSV

2023-12-07 09:07:58

2021-10-21 06:52:17

ZooKeeper分布式配置

2018-09-26 11:02:46

微服务架构组件

2022-08-18 20:45:30

HTTP协议数据

2021-04-11 08:30:40

VRAR虚拟现实技术

2021-11-10 07:47:48

Traefik边缘网关

2021-12-13 10:43:45

HashMapJava集合容器

2023-11-22 07:54:33

Xargs命令Linux
点赞
收藏

51CTO技术栈公众号