1. 前言
本文主要记录一下Spring中bean的生命周期,即从bean实例化到最终销毁的过程中一些重要的节点,以及Spring提供的在bean这些节点过程中的一些操作。
2. Bean生命周期
如下图所示:
2.1 BeanFactoryPostProcessor
该接口有如下方法:
void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;
该接口的实现类最先被实例化,它是一个spring容器级别的后置处理器,尽管此时所有的bean都没有被实例化出来,但是bean的配置文件已经被分析完,所有的bean的定义信息(spring中使用类BeanDefinition表示)已经加载,可以通过ConfigurableListableBeanFactory # getBeanDefinition(String beanName)
获取,获取到BeanDefiniton意味着你可以修改bean的属性了。
BeanFactoryPostProcessor 的一些实现类
- 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
- 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的一些实现类
- AutowiredAnnotationBeanPostProcessor
用于使用@Autowired
和@Value
注解的解析 - RequiredAnnotationBeanPostProcessor
用于使用@Require
注解的解析 - CommonAnnotationBeanPostProcessor
用于使用@Resource
,@PostConstruct
,@PreDestroy
注解的解析。
上面三类在在基于注解的方式装配bean时经常使用,当在xml文件里配置
<context:annotation-config />
时,spring会创建上面三类PostProcessor,同样输入context空间,参考 附录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的类
- 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