spring生命周期管理

1. 前言

本文主要记录一下Spring中bean的生命周期,即从bean实例化到最终销毁的过程中一些重要的节点,以及Spring提供的在bean这些节点过程中的一些操作。

2. Bean生命周期

如下图所示:

Spring生命周期.png

2.1 BeanFactoryPostProcessor

该接口有如下方法:

void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;

该接口的实现类最先被实例化,它是一个spring容器级别的后置处理器,尽管此时所有的bean都没有被实例化出来,但是bean的配置文件已经被分析完,所有的bean的定义信息(spring中使用类BeanDefinition表示)已经加载,可以通过ConfigurableListableBeanFactory # getBeanDefinition(String beanName)获取,获取到BeanDefiniton意味着你可以修改bean的属性了。

BeanFactoryPostProcessor 的一些实现类

  1. PropertyPlaceholderConfigurer
    这个很多人用过,从key=value形式的配置文件中加载配置信息,并通过${propertyName}的形式设置到bean属性上,如下面这种形式:
<context:property-placeholder
         location="classpath:db.properties"
          file-encoding="utf8"
/>

<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="${db.driver}"/>
        <property name="url" value="${db.url}"/>
        <property name="username" value="${db.user}"/>
        <property name="password" value="${db.password}"/>
</bean>

注:上面使用了简易的形式<context:property-placeholder .../>, 这种情况下会创建PropertyPlaceholderConfigurer对应的bean。关于<context>命名空间,请看 附录1

  1. PropertyOverrideConfigurer
    这个和 PropertyPlaceholderConfigurer一样将bean的属性卸载properties文件中,用法如下:
首先假设在db.properties中如下内容:
dataSource.username=root
dataSource.password=root
同样是name=value的形式,但是name必须是<beanName>.<beanPropertyName>格式

然后在applicationContext.xml中如下定义bean即可:
 <!--这个命名空间会加载PropertyOverrideConfigurer-->
<context:property-override location="db.properties"/>

<!--bean id需要和db.properties中beanName一样-->
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">   
</bean>
PropertyOverrideConfigurer这个后置处理器会根据db.properties中dataSource这个bean的名字找到属性,然后填充到dataSource这个bean中

2.2.2 BeanPostProcessor

BeanPostProcessor有两个接口方法:

Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException;
Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException;

BeanPostProcessor在普通bean实例化之前就就创建了,但是他的两个接口方法在却在普通bean实例化(即创建bean类的对象),通过setter完成了属性注入之后调用,这两个方法在bean初始化前后调用(bean初始化是指调用xml中bean定义中init-method指定的方法,或者bean实现了InitializingBean接口,调用接口方法),详细的可见上图。

BeanPostProcessor的一些实现类

  1. AutowiredAnnotationBeanPostProcessor
    用于使用@Autowired@Value注解的解析
  2. RequiredAnnotationBeanPostProcessor
    用于使用@Require注解的解析
  3. CommonAnnotationBeanPostProcessor
    用于使用@Resource, @PostConstruct,@PreDestroy注解的解析。

上面三类在在基于注解的方式装配bean时经常使用,当在xml文件里配置<context:annotation-config />时,spring会创建上面三类PostProcessor,同样输入context空间,参考 附录1

  1. ScheduledAnnotationBeanPostProcessor
    用于使用了@Scheduled注解的方法的解析,在xml中添加<task:annotation-driven>时会创建这个PostProcessor, 如下例所示:
 <beans>
   <task:annotation-driven scheduler="taskScheduler"/>
   
   <task:scheduler id="taskScheduler" pool-size="42"/>
   
   <task:scheduled-tasks scheduler="taskScheduler">
         <task:scheduled ref="myTask" method="work" fixed-rate="1000"/>
   </task:scheduled-tasks>
   
   <bean id="myTask" class="com.foo.MyTask"/>
</beans>
这个例子来自于spring代码@EnableScheduling注释

2.2.3 ApplicationContextAware

它只有一个接口方法setApplicationContext(ApplicationContext apc), 实现它的bean而言,apc即该bean运行时所处的applicationContext,上图没有画出它的调用时机,它在BeanFactoryAware # setBeanFactory之后调用。

实现ApplicationContextAware的类

  1. EventListenerMethodProcessor
    spring中事件机制中,用来处理bean中使用了@EventListener标注成事件接收这个的method。
    关于事件机制参考spring 事件机制

附录1 - context命名空间

spring-context的jar包中META-INF目录下有文件‘spring.handlers’中有这样一行http\://www.springframework.org/schema/context=org.springframework.context.config.ContextNamespaceHandler, context命名空间加载了类ContextNamespaceHandler,下面是这个类的init方法:

@Override
    public void init() {
        registerBeanDefinitionParser("property-placeholder", new PropertyPlaceholderBeanDefinitionParser());
        registerBeanDefinitionParser("property-override", new PropertyOverrideBeanDefinitionParser());
        registerBeanDefinitionParser("annotation-config", new AnnotationConfigBeanDefinitionParser());
        registerBeanDefinitionParser("component-scan", new ComponentScanBeanDefinitionParser());
        registerBeanDefinitionParser("load-time-weaver", new LoadTimeWeaverBeanDefinitionParser());
        registerBeanDefinitionParser("spring-configured", new SpringConfiguredBeanDefinitionParser());
        registerBeanDefinitionParser("mbean-export", new MBeanExportBeanDefinitionParser());
        registerBeanDefinitionParser("mbean-server", new MBeanServerBeanDefinitionParser());
    }

包含了context这个命名空间下所有子元素对应的Parser, 这写parser负责解析xml元素,然后创建对应BeanDefinition

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,558评论 19 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 47,084评论 6 342
  • 什么是Spring Spring是一个开源的Java EE开发框架。Spring框架的核心功能可以应用在任何Jav...
    jemmm阅读 16,685评论 1 133
  • 最近部门有三位同事要离职,其中一位与我同期入职,本来部门人就不多,如此规模的离职让我感到很受伤。 有人会说,同事嘛...
    日月空灏瑜阅读 4,262评论 0 1
  • 《心灵地图》是龙门院推出的人物访谈记录,由梁龙蜀担任采访者,选取在教育、文化、科学、商业等领域里做出个人特色作品,...
    龙门院的龙公子阅读 7,038评论 2 1