Java Spring SPringBoot

前言

在实际工作中总是需要在项目启动时做一些初始化的操作,比如初始化线程池、提前加载好加密证书…….
有哪些手段在Spring Boot 项目启动的时候做一些事情?方法有很多种,下面介绍几种常见的方法。

1、监听容器刷新完成扩展点ApplicationListener<ContextRefreshedEvent>

ApplicationContext事件机制是观察者设计模式实现的,通过ApplicationEventApplicationListener这两个接口实现ApplicationContext的事件机制。
Spring中一些内置的事件如下:

  1. ContextRefreshedEventApplicationContext 被初始化或刷新时,该事件被发布。这也可以在 ConfigurableApplicationContext接口中使用 refresh() 方法来发生。此处的初始化是指:所有的Bean被成功装载,后处理Bean被检测并激活,所有Singleton Bean 被预实例化,ApplicationContext容器已就绪可用。
  2. ContextStartedEvent:当使用 ConfigurableApplicationContextApplicationContext子接口)接口中的 start() 方法启动 ApplicationContext 时,该事件被发布。可以调查数据库,或者可以在接受到这个事件后重启任何停止的应用程序。
  3. ContextStoppedEvent:当使用 ConfigurableApplicationContext 接口中的 stop() 停止 ApplicationContext 时,发布这个事件。可以在接受到这个事件后做必要的清理的工作。
  4. ContextClosedEvent:当使用 ConfigurableApplicationContext 接口中的 close() 方法关闭 ApplicationContext 时,该事件被发布。一个已关闭的上下文到达生命周期末端;它不能被刷新或重启。
  5. RequestHandledEvent:这是一个 web-specific 事件,告诉所有 bean HTTP 请求已经被服务。只能应用于使用DispatcherServlet的Web应用。在使用Spring作为前端的MVC控制器时,当Spring处理用户请求结束后,系统会自动触发该事件。

好了,了解上面这些内置事件后,可以监听ContextRefreshedEvent在Spring Boot 启动时完成一些操作,代码如下:

  1. @Component
  2. public class TestApplicationListener implements ApplicationListener<ContextRefreshedEvent>{
  3. @Override
  4. public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {
  5. System.out.println(contextRefreshedEvent);
  6. System.out.println("TestApplicationListener............................");
  7. }
  8. }

高级玩法

可以自定事件完成一些特定的需求,比如:邮件发送成功之后,做一些业务处理。

自定义EmailEvent

代码如下:

  1. public class EmailEvent extends ApplicationEvent{
  2. private String address;
  3. private String text;
  4. public EmailEvent(Object source, String address, String text){
  5. super(source);
  6. this.address = address;
  7. this.text = text;
  8. }
  9. public EmailEvent(Object source) {
  10. super(source);
  11. }
  12. //......address和text的setter、getter
  13. }

自定义监听器

代码如下:

  1. public class EmailNotifier implements ApplicationListener{
  2. public void onApplicationEvent(ApplicationEvent event) {
  3. if (event instanceof EmailEvent) {
  4. EmailEvent emailEvent = (EmailEvent)event;
  5. System.out.println("邮件地址:" + emailEvent.getAddress());
  6. System.our.println("邮件内容:" + emailEvent.getText());
  7. } else {
  8. System.our.println("容器本身事件:" + event);
  9. }
  10. }
  11. }

发送邮件后,触发事件

代码如下:

  1. public class SpringTest {
  2. public static void main(String args[]){
  3. ApplicationContext context = new ClassPathXmlApplicationContext("bean.xml");
  4. //创建一个ApplicationEvent对象
  5. EmailEvent event = new EmailEvent("hello","abc@163.com","This is a test");
  6. //主动触发该事件
  7. context.publishEvent(event);
  8. }
  9. }

2、SpringBoot的CommandLineRunner接口

当容器初始化完成之后会调用CommandLineRunner中的run()方法,同样能够达到容器启动之后完成一些事情。这种方式和ApplicationListener相比更加灵活,如下:

  • 不同的CommandLineRunner实现可以通过@Order()指定执行顺序
  • 可以接收从控制台输入的参数。

下面自定义一个实现类,代码如下:

  1. @Component
  2. @Slf4j
  3. public class CustomCommandLineRunner implements CommandLineRunner {
  4. /**
  5. * @param args 接收控制台传入的参数
  6. */
  7. @Override
  8. public void run(String... args) throws Exception {
  9. log.debug("从控制台接收参数>>>>"+ Arrays.asList(args));
  10. }
  11. }

运行这个jar,命令如下:

  1. java -jar demo.jar aaa bbb ccc

以上命令中传入了三个参数,分别是aaa、bbb、ccc,这三个参数将会被run()方法接收到。如下图:
2021-08-31-14-24-10-358411.png

源码分析

Spring Boot 加载上下文的入口在org.springframework.context.ConfigurableApplicationContext()这个方法中,如下图:
2021-08-31-14-24-10-566406.png
调用CommandLineRunnercallRunners(context, applicationArguments);这个方法中执行,源码如下图:
2021-08-31-14-24-10-698403.png

3、SpringBoot的ApplicationRunner接口

ApplicationRunnerCommandLineRunner都是Spring Boot 提供的,相对于CommandLineRunner来说对于控制台传入的参数封装更好一些,可以通过键值对来获取指定的参数,比如--version=2.1.0
此时运行这个jar命令如下:

  1. java -jar demo.jar --version=2.1.0 aaa bbb ccc

以上命令传入了四个参数,一个键值对version=2.1.0,另外三个是分别是aaa、bbb、ccc。
同样可以通过@Order()指定优先级,如下代码:

  1. @Component
  2. @Slf4j
  3. public class CustomApplicationRunner implements ApplicationRunner {
  4. @Override
  5. public void run(ApplicationArguments args) throws Exception {
  6. log.debug("控制台接收的参数:{},{},{}",args.getOptionNames(),args.getNonOptionArgs(),args.getSourceArgs());
  7. }
  8. }

通过以上命令运行,结果如下图:
2021-08-31-14-24-11-141404.png

源码分析

CommandLineRunner一样,同样在callRunners()这个方法中执行,源码如下图:
2021-08-31-14-24-11-270407.png

4、@PostConstruct注解

前三种针对的是容器的初始化完成之后做的一些事情,@PostConstruct这个注解是针对Bean的初始化完成之后做一些事情,比如注册一些监听器…
@PostConstruct注解一般放在Bean的方法上,一旦Bean初始化完成之后,将会调用这个方法,代码如下:

  1. @Component
  2. @Slf4j
  3. public class SimpleExampleBean {
  4. @PostConstruct
  5. public void init(){
  6. log.debug("Bean初始化完成,调用...........");
  7. }
  8. }

5、@Bean注解中指定初始化方法

这种方式和@PostConstruct比较类似,同样是指定一个方法在Bean初始化完成之后调用。
新建一个Bean,代码如下:

  1. @Slf4j
  2. public class SimpleExampleBean {
  3. public void init(){
  4. log.debug("Bean初始化完成,调用...........");
  5. }
  6. }

在配置类中通过@Bean实例化这个Bean,不过@Bean中的initMethod这个属性需要指定初始化之后需要执行的方法,如下:

  1. @Bean(initMethod = "init")
  2. public SimpleExampleBean simpleExampleBean(){
  3. return new SimpleExampleBean();
  4. }

6、InitializingBean接口

InitializingBean的用法基本上与@PostConstruct一致,只不过相应的Bean需要实现afterPropertiesSet方法,代码如下:

  1. @Slf4j
  2. @Component
  3. public class SimpleExampleBean implements InitializingBean {
  4. @Override
  5. public void afterPropertiesSet() {
  6. log.debug("Bean初始化完成,调用...........");
  7. }
  8. }