博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
spring的InitializingBean的 afterPropertiesSet 方法 和 init-method配置的区别联系
阅读量:6961 次
发布时间:2019-06-27

本文共 4895 字,大约阅读时间需要 16 分钟。

   InitializingBean  

Spirng的InitializingBean为bean提供了定义初始化方法的方式。InitializingBean是一个接口,它仅仅包含一个方法:afterPropertiesSet()。  

Bean实现这个接口,在afterPropertiesSet()中编写初始化代码:    

package research.spring.beanfactory.ch4; import org.springframework.beans.factory.InitializingBean;      class      LifeCycleBean implements InitializingBean{      void      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">          

        >           

<          /beans                >           

编写测试程序进行测试:    

package research.spring.beanfactory.ch4; import org.springframework.beans.factory.xml.XmlBeanFactory; import org.springframework.core.io.ClassPathResource;      public       class      LifeCycleTest {      static       void      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的初始化方法。          protected           void          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-method           if          (mergedBeanDefinition!=          null          &&mergedBeanDefinition.getInitMethodName() !=          null          ) {       

//调用invokeCustomInitMethod方法来执行init-method定义的方法

invokeCustomInitMethod(beanName, bean, mergedBeanDefinition.getInitMethodName());

} }

//           执行一个bean定义的init-method方法           protected           void          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 +"'");

}

//           判断方法是否是public           if          (!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上。

转载于:https://www.cnblogs.com/xuxiuxiu/p/6829266.html

你可能感兴趣的文章
yii2使用多个数据库的案例
查看>>
[LintCode/LeetCode] Search Insert Position
查看>>
Node处理http跨域请求
查看>>
Gradle for Android 第五篇( 多模块构建 )
查看>>
框架 也可以这样用 easy
查看>>
LCUI 1.1.0 Beta 发布,C 的图形界面库
查看>>
Base64
查看>>
MariaDB(mysql)之半同步复制及复制过滤器的使用
查看>>
WCF Ria Service 理解制图版本【待续】
查看>>
【IOS】高仿糗事百科客户端(基本实现了除注册,评论之外的功能)
查看>>
puppet之模块详解
查看>>
shell脚本
查看>>
python 模块
查看>>
Flash Builder生成asdoc格式的帮助文档
查看>>
分享实录 | 第四范式程晓澄:机器学习在推荐系统中的应用
查看>>
测试服务器响应时间
查看>>
xm console无法联接guest问题的解决
查看>>
一步一步SharePoint 2007之一:安装SharePoint
查看>>
Server Develop (七) Linux 守护进程
查看>>
Android requires compiler compliance level 5.0. Please fix project properties.错误
查看>>