Spirng的InitializingBean为bean提供了定义初始化方法的方式。InitializingBean是一个接口,它仅仅包含一个方法:afterPropertiesSet()。spring
Bean实现这个接口,在afterPropertiesSet()中编写初始化代码:测试
package research.spring.beanfactory.ch4; import org.springframework.beans.factory.InitializingBean;publicclass LifeCycleBean implements InitializingBean{publicvoid afterPropertiesSet() throws Exception { System.out.println("LifeCycleBean initializing..."); } }
在xml配置文件中并不须要对bean进行特殊的配置:spa
xml version="1.0" encoding="UTF-8"?>DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd"><beans><bean name="lifeBean" class="research.spring.beanfactory.ch4.LifeCycleBean">bean>beans>
编写测试程序进行测试:debug
package research.spring.beanfactory.ch4; import org.springframework.beans.factory.xml.XmlBeanFactory; import org.springframework.core.io.ClassPathResource;publicclass LifeCycleTest {publicstaticvoid main(String[] args) { XmlBeanFactory factory=new XmlBeanFactory(new ClassPathResource(
"research/spring/beanfactory/ch4/context.xml")); factory.getBean("lifeBean"); } }code
运行上面的程序咱们会看到:“LifeCycleBean initializing...”,这说明bean的afterPropertiesSet已经被Spring调用了。orm
Spring在设置完一个bean全部的合做者后,会检查bean是否实现了InitializingBean接口,若是实现就调用bean的afterPropertiesSet方法。xml
SHAPE /* MERGEFORMAT对象
Spring虽然能够经过InitializingBean完成一个bean初始化后对这个bean的回调,可是这种方式要求bean实现 InitializingBean接口。一但bean实现了InitializingBean接口,那么这个bean的代码就和Spring耦合到一块儿了。一般状况下我不鼓励bean直接实现InitializingBean,可使用Spring提供的init-method的功能来执行一个bean 子定义的初始化方法。
写一个java class,这个类不实现任何Spring的接口。定义一个没有参数的方法init()。
package research.spring.beanfactory.ch4;publicclass LifeCycleBean{publicvoid init(){ System.out.println("LifeCycleBean.init..."); } }
在Spring中配置这个bean:
xml version="1.0" encoding="UTF-8"?>DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd"><beans><bean name="lifeBean" class="research.spring.beanfactory.ch4.LifeCycleBean"
init-method="init">bean>beans>
当Spring实例化lifeBean时,你会在控制台上看到” LifeCycleBean.init...”。
Spring要求init-method是一个无参数的方法,若是init-method指定的方法中有参数,那么Spring将会抛出java.lang.NoSuchMethodException
init-method指定的方法能够是public、protected以及private的,而且方法也能够是final的。
init-method指定的方法能够是声明为抛出异常的,就像这样:
final protected void init() throws Exception{
System.out.println("init method...");
if(true) throw new Exception("init exception");
}
若是在init-method方法中抛出了异常,那么Spring将停止这个Bean的后续处理,而且抛出一个org.springframework.beans.factory.BeanCreationException异常。
InitializingBean和init-method能够一块儿使用,Spring会先处理InitializingBean再处理init-method。
org.springframework.beans.factory.support. AbstractAutowireCapableBeanFactory完成一个Bean初始化方法的调用工做。 AbstractAutowireCapableBeanFactory是XmlBeanFactory的超类,再 AbstractAutowireCapableBeanFactory的invokeInitMethods方法中实现调用一个Bean初始化方法:
org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.java:
//……//在一个bean的合做者设备完成后,执行一个bean的初始化方法。protectedvoid invokeInitMethods(String beanName, Object bean, RootBeanDefinition mergedBeanDefinition)
throws Throwable {//判断bean是否实现了InitializingBean接口if (bean instanceof InitializingBean) {if (logger.isDebugEnabled()) { logger.debug("Invoking afterPropertiesSet() on bean with name '"+ beanName +"'"); }//调用afterPropertiesSet方法((InitializingBean) bean).afterPropertiesSet(); }//判断bean是否认义了init-methodif(mergedBeanDefinition!=null&&mergedBeanDefinition.getInitMethodName() !=null) {//调用invokeCustomInitMethod方法来执行init-method定义的方法invokeCustomInitMethod(beanName, bean, mergedBeanDefinition.getInitMethodName()); } }//执行一个bean定义的init-method方法protectedvoid invokeCustomInitMethod(String beanName, Object bean, String initMethodName) throws Throwable {if (logger.isDebugEnabled()) { logger.debug("Invoking custom init method '"+ initMethodName +"' on bean with name '"+ beanName +"'"); }//使用方法名,反射Method对象Method initMethod = BeanUtils.findMethod(bean.getClass(), initMethodName, null);if (initMethod ==null) {thrownew NoSuchMethodException(
"Couldn't find an init method named '"+ initMethodName +"' on bean with name '"+ beanName +"'"); }//判断方法是不是publicif (!Modifier.isPublic(initMethod.getModifiers())) {//设置accessible为true,能够访问private方法。 initMethod.setAccessible(true); }try {//反射执行这个方法initMethod.invoke(bean, (Object[]) null); }catch (InvocationTargetException ex) {throw ex.getTargetException(); } }//………..
经过分析上面的源代码咱们能够看到,init-method是经过反射执行的,而afterPropertiesSet是直接执行的。因此 afterPropertiesSet的执行效率比init-method要高,不过init-method消除了bean对Spring依赖。在实际使用时我推荐使用init-method。
须要注意的是Spring老是先处理bean定义的InitializingBean,而后才处理init-method。若是在Spirng处理InitializingBean时出错,那么Spring将直接抛出异常,不会再继续处理init-method。
若是一个bean被定义为非单例的,那么afterPropertiesSet和init-method在bean的每个实例被建立时都会执行。单例 bean的afterPropertiesSet和init-method只在bean第一次被实例时调用一次。大多数状况下 afterPropertiesSet和init-method都应用在单例的bean上。