查看: 301|回复: 0

[Java代码] 谈谈我对Spring Bean 生命周期的理解

发表于 2018-4-14 08:00:02

前言

Spring的ioc容器功能非常强大,负责Spring的Bean的创建和管理等功能。而Spring 的bean是整个Spring应用中很重要的一部分,了解Spring Bean的生命周期对我们了解整个spring框架会有很大的帮助。

BeanFactory和ApplicationContext是Spring两种很重要的容器,前者提供了最基本的依赖注入的支持,而后者在继承前者的基础进行了功能的拓展,例如增加了事件传播,资源访问和国际化的消息访问等功能。本文主要介绍了ApplicationContext和BeanFactory两种容器的Bean的生命周期。

首先看下生命周期图:

再谈生命周期之前有一点需要先明确:

Spring 只帮我们管理单例模式 Bean 的 完整 生命周期,对于 prototype 的 bean ,Spring 在创建好交给使用者之后则不会再管理后续的生命周期。

注解方式

在 bean 初始化时会经历几个阶段,首先可以使用注解 @PostConstruct , @PreDestroy 来在 bean 的创建和销毁阶段进行调用:

  1. @Component
  2. public class AnnotationBean {
  3. private final static Logger LOGGER = LoggerFactory.getLogger(AnnotationBean.class);
  4. @PostConstruct
  5. public void start(){
  6. LOGGER.info("AnnotationBean start");
  7. }
  8. @PreDestroy
  9. public void destroy(){
  10. LOGGER.info("AnnotationBean destroy");
  11. }
  12. }
复制代码

InitializingBean, DisposableBean 接口

还可以实现 InitializingBean,DisposableBean 这两个接口,也是在初始化以及销毁阶段调用:

  1. @Service
  2. public class SpringLifeCycleService implements InitializingBean,DisposableBean{
  3. private final static Logger LOGGER = LoggerFactory.getLogger(SpringLifeCycleService.class);
  4. @Override
  5. public void afterPropertiesSet() throws Exception {
  6. LOGGER.info("SpringLifeCycleService start");
  7. }
  8. @Override
  9. public void destroy() throws Exception {
  10. LOGGER.info("SpringLifeCycleService destroy");
  11. }
  12. }
复制代码

自定义初始化和销毁方法

也可以自定义方法用于在初始化、销毁阶段调用:

  1. @Configuration
  2. public class LifeCycleConfig {
  3. @Bean(initMethod = "start", destroyMethod = "destroy")
  4. public SpringLifeCycle create(){
  5. SpringLifeCycle springLifeCycle = new SpringLifeCycle() ;
  6. return springLifeCycle ;
  7. }
  8. }
  9. public class SpringLifeCycle{
  10. private final static Logger LOGGER = LoggerFactory.getLogger(SpringLifeCycle.class);
  11. public void start(){
  12. LOGGER.info("SpringLifeCycle start");
  13. }
  14. public void destroy(){
  15. LOGGER.info("SpringLifeCycle destroy");
  16. }
  17. }
复制代码

以上是在 SpringBoot 中可以这样配置,如果是原始的基于 XML 也是可以使用:

  1. <bean class="com.crossoverjie.spring.SpringLifeCycle" init-method="start" destroy-method="destroy">
  2. </bean>
复制代码

来达到同样的效果。

实现 *Aware 接口

*Aware 接口可以用于在初始化 bean 时获得 Spring 中的一些对象,如获取 Spring 上下文 等。

  1. @Component
  2. public class SpringLifeCycleAware implements ApplicationContextAware {
  3. private final static Logger LOGGER = LoggerFactory.getLogger(SpringLifeCycleAware.class);
  4. private ApplicationContext applicationContext ;
  5. @Override
  6. public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
  7. this.applicationContext = applicationContext ;
  8. LOGGER.info("SpringLifeCycleAware start");
  9. }
  10. }
复制代码

这样在 springLifeCycleAware 这个 bean 初始化会就会调用 setApplicationContext 方法,并可以获得 applicationContext 对象。

BeanPostProcessor 增强处理器

实现 BeanPostProcessor 接口,Spring 中所有 bean 在做初始化时都会调用该接口中的两个方法,可以用于对一些特殊的 bean 进行处理:

  1. @Component
  2. public class SpringLifeCycleProcessor implements BeanPostProcessor {
  3. private final static Logger LOGGER = LoggerFactory.getLogger(SpringLifeCycleProcessor.class);
  4. /**
  5. * 预初始化 初始化之前调用
  6. * @param bean
  7. * @param beanName
  8. * @return
  9. * @throws BeansException
  10. */
  11. @Override
  12. public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
  13. if ("annotationBean".equals(beanName)){
  14. LOGGER.info("SpringLifeCycleProcessor start beanName={}",beanName);
  15. }
  16. return bean;
  17. }
  18. /**
  19. * 后初始化 bean 初始化完成调用
  20. * @param bean
  21. * @param beanName
  22. * @return
  23. * @throws BeansException
  24. */
  25. @Override
  26. public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
  27. if ("annotationBean".equals(beanName)){
  28. LOGGER.info("SpringLifeCycleProcessor end beanName={}",beanName);
  29. }
  30. return bean;
  31. }
  32. }
复制代码

执行之后观察结果:

  1. 018-03-21 00:40:24.856 [restartedMain] INFO c.c.s.p.SpringLifeCycleProcessor - SpringLifeCycleProcessor start beanName=annotationBean
  2. 2018-03-21 00:40:24.860 [restartedMain] INFO c.c.spring.annotation.AnnotationBean - AnnotationBean start
  3. 2018-03-21 00:40:24.861 [restartedMain] INFO c.c.s.p.SpringLifeCycleProcessor - SpringLifeCycleProcessor end beanName=annotationBean
  4. 2018-03-21 00:40:24.864 [restartedMain] INFO c.c.s.aware.SpringLifeCycleAware - SpringLifeCycleAware start
  5. 2018-03-21 00:40:24.867 [restartedMain] INFO c.c.s.service.SpringLifeCycleService - SpringLifeCycleService start
  6. 2018-03-21 00:40:24.887 [restartedMain] INFO c.c.spring.SpringLifeCycle - SpringLifeCycle start
  7. 2018-03-21 00:40:25.062 [restartedMain] INFO o.s.b.d.a.OptionalLiveReloadServer - LiveReload server is running on port 35729
  8. 2018-03-21 00:40:25.122 [restartedMain] INFO o.s.j.e.a.AnnotationMBeanExporter - Registering beans for JMX exposure on startup
  9. 2018-03-21 00:40:25.140 [restartedMain] INFO com.crossoverjie.Application - Started Application in 2.309 seconds (JVM running for 3.681)
  10. 2018-03-21 00:40:25.143 [restartedMain] INFO com.crossoverjie.Application - start ok!
  11. 2018-03-21 00:40:25.153 [Thread-8] INFO o.s.c.a.AnnotationConfigApplicationContext - Closing org.springframework.context.annotation.AnnotationConfigApplicationContext@3913adad: startup date [Wed Mar 21 00:40:23 CST 2018]; root of context hierarchy
  12. 2018-03-21 00:40:25.155 [Thread-8] INFO o.s.j.e.a.AnnotationMBeanExporter - Unregistering JMX-exposed beans on shutdown
  13. 2018-03-21 00:40:25.156 [Thread-8] INFO c.c.spring.SpringLifeCycle - SpringLifeCycle destroy
  14. 2018-03-21 00:40:25.156 [Thread-8] INFO c.c.s.service.SpringLifeCycleService - SpringLifeCycleService destroy
  15. 2018-03-21 00:40:25.156 [Thread-8] INFO c.c.spring.annotation.AnnotationBean - AnnotationBean destroy
复制代码

直到 Spring 上下文销毁时则会调用自定义的销毁方法以及实现了 DisposableBean 的 destroy() 方法。

总结

以上所述是小编给大家介绍的Spring Bean 生命周期,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对程序员之家网站的支持!



回复

使用道具 举报