?
?
InitializingBean??? Spirng的InitializingBean為bean提供了定義初始化方法的方式。InitializingBean是一個接口,它僅僅包含一個方法:afterPropertiesSet()。
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進行特殊的配置:
?
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>
?
編寫測試程序進行測試:
?
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");}}
??? 運行上面的程序我們會看到:“LifeCycleBean initializing...”,這說明bean的afterPropertiesSet已經被Spring調用了。
?
??? Spring在設置完一個bean所有的合作者后,會檢查bean是否實現了InitializingBean接口,如果實現就調用bean的afterPropertiesSet方法。
?SHAPE? \* MERGEFORMAT
? 裝配bean的合作者 ? |
? 查看bean是否實現InitializingBean接口 ? |
? 調用afterPropertiesSet方法 ? |
?
??? 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是直接執行的。所
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
