Spring启动时实现初始化有哪些方式?
作者:pbxs 发布时间:2023-09-20 18:25:57
一、Spring启动时实现初始化的几种方式
准确的说是spring容器实例化完成后,几种初始化的方式。为什么这么说呢?下看面示例:
@Slf4j
@Component
public class InitBeanDemo {
@Autowired
private Environment env;
public InitBeanDemo() {
log.info("DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
示例是在bean的构造方法里做一些初始化的工作,示例比较简单只做了日志打印。是理想很丰满,现实很骨感,报错了:Constructor threw exception; nested exception is java.lang.NullPointerException。
原因是,Environment尚未初始化完成。
接下来我们来探索一下 有哪些初始化方式能满足上面示例的需求。
二、构造方法里初始化
可以正常运行,在所有初始化方式里执行时机最早。原理是在InitBeanDemo实例化前就实例化了Environment。
@Component
public class InitBeanDemo {
private final Environment env;
@Autowired
public InitBeanDemo (Environment environment) {
this.env = environment;
log.info("Constructor DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("Constructor ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
}
三、常规三件套
常规三件套:@PostConstruct、InitializingBean、initMethod。 如果你愿意的话,三种方式可以在同一个Bean下同时使用,执行的优先级@PostConstruct > InitializingBean > initMethod。
@PostConstruct注解
在一个可以被扫描到Bean里,添加一个public void xxx()方法并加上@PostConstruct注解,方法里编写需要初始化的逻辑。
同一个应用程序里可以有多个@PostConstruct注解,同一个Bean里也可以有多个@PostConstruct注解。
@Slf4j
@Component
public class InitBeanDemo {
@Autowired
private Environment env;
@PostConstruct
public void init() {
log.info("@PostConstruct DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("@PostConstruct ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
实现InitializingBean接口
实现InitializingBean接口,在afterPropertiesSet() 方法里编写需要初始化的逻辑。
同一个应用程序里可以有多个实现InitializingBean接口的类,执行时机会按类名的自然顺序排序。
@Slf4j
@Component
public class InitBeanDemo implements InitializingBean {
@Autowired
private Environment env;
@Override
public void afterPropertiesSet() throws Exception {
log.info("InitializingBean DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("InitializingBean ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
}
指定Bean的initMethod方法
使用@Bean注解的initMethod属性可用于Bean的初始化后执行的方法。initMethod必须是public void 的无参构造方法。
@Slf4j
public class InitBeanDemo implements InitializingBean {
@Autowired
private Environment env;
public void initMethod() {
log.info("initMethod DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("initMethod ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
@Configuration
public class InitBeanConfig {
@Bean(initMethod="initMethod")
public InitBeanDemo initBeanDemo () {
return new InitBeanDemo();
}
}
等同于 在XML 配置中的init-method属性:
<bean id="initBeanDemo" class="com.xxx.InitBeanDemo" init-method="initMethod"></bean>
四、自定义ApplicationListener监听
两种方式,一种实现接口,另一种使用注解。
实现ApplicationListener接口
监听ContextRefreshedEvent事件。
@Slf4j
@Component
public class InitBeanDemo implements ApplicationListener<ContextRefreshedEvent>{
@Autowired
private Environment env;
@Override
public void onApplicationEvent(ContextRefreshedEvent event) {
log.info("ApplicationListener DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("ApplicationListener ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
}
@EventListener注释
方法参数里指定ContextRefreshedEvent事件。
@Slf4j
@Component
public class InitBeanDemo {
@Autowired
private Environment env;
@EventListener
public void onApplicationEvent2(ContextRefreshedEvent event) {
log.info("@EventListener DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("@EventListener ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
}
五、Spring Boot提供的初始化接口
ApplicationRunner接口
@Slf4j
@Component
public class InitBeanDemo implements ApplicationRunner {
@Autowired
private Environment env;
@Override
public void run(ApplicationArguments args) throws Exception {
log.info("ApplicationRunner: {}", args);
log.info("ApplicationRunner: {}", args.getOptionNames());
log.info("ApplicationRunner DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("ApplicationRunner ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
}
CommandLineRunner接口
可以在同一个应用程序上下文中定义多个CommandLineRunner bean,并且可以使用@Ordered接口或@Order注释进行排序。
@Slf4j
@Component
public class InitBeanDemo implements CommandLineRunner {
@Autowired
private Environment env;
@Override
public void run(String... args) throws Exception {
log.info("CommandLineRunner: {}", args);
log.info("CommandLineRunner DefaultProfiles: {}", Arrays.asList(env.getDefaultProfiles()));
log.info("CommandLineRunner ActiveProfiles: {}", Arrays.asList(env.getActiveProfiles()));
}
}
在同一个Bean里使用以上初始化方式的执行先后顺序
在同一个Bean里使用以上初始化方式,运行的日志片段:
2021-06-07 11:24:41|INFO |main|c.c.s.s.t.ConstructorInitDemo|Constructor DefaultProfiles: [default]
2021-06-07 11:24:41|INFO |main|c.c.s.s.t.ConstructorInitDemo|Constructor ActiveProfiles: [sit]
2021-06-07 11:24:42|INFO |main|c.c.s.s.test.InitBeanDemo|@PostConstruct DefaultProfiles: [default]
2021-06-07 11:24:42|INFO |main|c.c.s.s.test.InitBeanDemo|@PostConstruct ActiveProfiles: [sit]
2021-06-07 11:24:42|INFO |main|c.c.s.s.test.InitBeanDemo|InitializingBean DefaultProfiles: [default]
2021-06-07 11:24:42|INFO |main|c.c.s.s.test.InitBeanDemo|InitializingBean ActiveProfiles: [sit]
2021-06-07 11:24:42|INFO |main|c.c.s.s.test.InitBeanDemo|initMethod DefaultProfiles: [default]
2021-06-07 11:24:42|INFO |main|c.c.s.s.test.InitBeanDemo|initMethod ActiveProfiles: [sit]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|@EventListener DefaultProfiles: [default]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|@EventListener ActiveProfiles: [sit]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|ApplicationListener DefaultProfiles: [default]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|ApplicationListener ActiveProfiles: [sit]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|ApplicationRunner: org.springframework.boot.DefaultApplicationArguments@68bef3df
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|ApplicationRunner: []
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|ApplicationRunner DefaultProfiles: [default]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|ApplicationRunner ActiveProfiles: [sit]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|CommandLineRunner: {}
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|CommandLineRunner DefaultProfiles: [default]
2021-06-07 11:24:44|INFO |main|c.c.s.s.test.InitBeanDemo|CommandLineRunner ActiveProfiles: [sit]
也即 整篇文章整理的先后顺序。
来源:https://blog.csdn.net/WLQ0621/article/details/117661645
猜你喜欢
- like模糊查询特殊字符报错转义处理方案1 <if test="projectName!
- 刚开始我以为熔断和降级是一体的,以为他们必须配合使用; 只不过名字不一样而已,但是当我经过思考过后,发现他们其实不是一个东西;降级什么是服务
- springboot和springmvc的区别spring boot 内嵌tomcat,Jetty和Undertow容器,可以直接运行起来,
- java 接口回调实例详解首先官方对接口回调的定义是这样的,所谓回调:就是A类中调用B类中的某个方法C,然后B类中反过来调用A类中的方法D,
- 什么是枚举?枚举是JDK5引入的新特性。在某些情况下,一个类的对象是固定的,就可以定义为枚举。在实际使用中,枚举类型也可以作为一种规范,保障
- 前言一直很好奇Android Root的原理,恰好最近碰到了一个跟Android默认带Root权限的问题,这里顺便记录一下Android系统
- java 基础之JavaBean属性命名规范问题JavaBean属性名要求:前两个字母要么都大写,要么都小写下面我们来找找如果不遵循这个规范
- 稳定度(稳定性)一个排序算法是稳定的,就是当有两个相等记录的关键字R和S,且在原本的列表中R出现在S之前,在排序过的列表中R也将会是在S之前
- 这篇文章主要介绍了Spring JDK * 实现过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要
- 前言用户注册功能是每一个系统的入口门面功能,很多人可能会以为很简单,不就是一个简单的CRUD吗?其实不然,要把前后端功能都做出来,页面跳转也
- 前言xxljob 是采用 java 开发的开源的任务调度系统,架构上分为调度管理器、执行器,目前除了官方提供的 java 执行器外,也有 g
- Java 异步实现的几种方式1. jdk1.8之前的Futurejdk并发包里的Future代表了未来的某个结果,当我们向线程池中提交任务的
- 目录一、导入依赖二、前端实现三、后台逻辑三、页面效果四、可能会遇到的问题一、导入依赖这里还是用了Apache的POI插件,现在一般的spri
- 废话不多说了,直接给大家贴代码了,具体代码如下所示:<update id="updateAuditStateAndType&
- using System.Xml;//初始化一个xml实例XmlDocument xml=new XmlDocument();//导入指定x
- 为什么需要ThreadLocalRandomjava.util.Random一直都是使用比较广泛的随机数生成工具类,而且java.lang.
- 本文实例为大家分享了Java实现寻找迷宫出路的具体代码,供大家参考,具体内容如下项目名称寻找迷宫出路项目描述给定一个自定义迷宫,0表示能通过
- 前言网上SSO的框架很多,此篇文章使用的是自写的SSO来实现简单的登录授权功能,目的在于扩展性,权限这方面,自写扩展性会好点。提示:以下是本
- JavaFX主要致力于富客户端开发,以弥补swing的缺陷,主要提供图形库与media库,支持audio,video,graphic,ani
- 一、简介在分布式系统中,由于服务数量巨多,为了方便服务配置文件统一管理,实时更新,所以需要分布式配置中心组件。在Spring Cloud中,