Spring啟動時(shí)實(shí)現(xiàn)初始化有哪些方式?
一、Spring啟動時(shí)實(shí)現(xiàn)初始化的幾種方式
準(zhǔn)確的說是spring容器實(shí)例化完成后,幾種初始化的方式。為什么這么說呢?下看面示例:
@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的構(gòu)造方法里做一些初始化的工作,示例比較簡單只做了日志打印。是理想很豐滿,現(xiàn)實(shí)很骨感,報(bào)錯(cuò)了:Constructor threw exception; nested exception is java.lang.NullPointerException。
原因是,Environment尚未初始化完成。
接下來我們來探索一下 有哪些初始化方式能滿足上面示例的需求。
二、構(gòu)造方法里初始化
可以正常運(yùn)行,在所有初始化方式里執(zhí)行時(shí)機(jī)最早。原理是在InitBeanDemo實(shí)例化前就實(shí)例化了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())); } }
三、常規(guī)三件套
常規(guī)三件套:@PostConstruct、InitializingBean、initMethod。 如果你愿意的話,三種方式可以在同一個(gè)Bean下同時(shí)使用,執(zhí)行的優(yōu)先級@PostConstruct > InitializingBean > initMethod。
@PostConstruct注解
在一個(gè)可以被掃描到Bean里,添加一個(gè)public void xxx()方法并加上@PostConstruct注解,方法里編寫需要初始化的邏輯。
同一個(gè)應(yīng)用程序里可以有多個(gè)@PostConstruct注解,同一個(gè)Bean里也可以有多個(gè)@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())); }
實(shí)現(xiàn)InitializingBean接口
實(shí)現(xiàn)InitializingBean接口,在afterPropertiesSet() 方法里編寫需要初始化的邏輯。
同一個(gè)應(yīng)用程序里可以有多個(gè)實(shí)現(xiàn)InitializingBean接口的類,執(zhí)行時(shí)機(jī)會按類名的自然順序排序。
@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的初始化后執(zhí)行的方法。initMethod必須是public void 的無參構(gòu)造方法。
@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監(jiān)聽
兩種方式,一種實(shí)現(xiàn)接口,另一種使用注解。
實(shí)現(xiàn)ApplicationListener接口
監(jiān)聽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注釋
方法參數(shù)里指定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接口
可以在同一個(gè)應(yīng)用程序上下文中定義多個(gè)CommandLineRunner bean,并且可以使用@Ordered接口或@Order注釋進(jìn)行排序。
@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())); } }
在同一個(gè)Bean里使用以上初始化方式的執(zhí)行先后順序
在同一個(gè)Bean里使用以上初始化方式,運(yùn)行的日志片段:
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]
也即 整篇文章整理的先后順序。
到此這篇關(guān)于Spring啟動時(shí)實(shí)現(xiàn)初始化有哪些方式?的文章就介紹到這了,更多相關(guān)Spring初始化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解mybatis.generator配上最新的mysql 8.0.11的一些坑
這篇文章主要介紹了詳解mybatis.generator配上最新的mysql 8.0.11的一些坑,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-10-10

springboot 項(xiàng)目使用jasypt加密數(shù)據(jù)源的方法

Spring Boot啟動過程(四)之Spring Boot內(nèi)嵌Tomcat啟動

關(guān)于Spring中@Transactional事務(wù)回滾的注意事項(xiàng)