首页 > 其他 > 详细

afterPropertiesSet方法和init-method区别和应用

时间:2020-09-25 19:07:37      阅读:265      评论:0      收藏:0      [点我收藏+]

InitializingBean

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

 

  1.  
    public interface InitializingBean
  2.  
    {
  3.  
    public abstract void afterPropertiesSet()
  4.  
    throws Exception;
  5.  
    }

 

用法示例:
Bean实现:
  1.  
    import org.springframework.beans.factory.InitializingBean;
  2.  
     
  3.  
    publicclass LifeCycleBean implements InitializingBean{
  4.  
    void afterPropertiesSet() throws Exception {
  5.  
    System.out.println("LifeCycleBean initializing...");
  6.  
    }
  7.  
    }

在xml配置文件中并不需要对bean进行特殊的配置:

  1.  
    <beans>
  2.  
    <bean name ="lifeBean" class ="com.spring.LifeCycleBean"></beans >
  3.  
    </beans>

编写测试程序进行测试:

 

  1.  
    import org.springframework.beans.factory.xml.XmlBeanFactory;
  2.  
    import org.springframework.core.io.ClassPathResource;
  3.  
     
  4.  
    public class LifeCycleTest {
  5.  
    public static void main(String[] args) {
  6.  
    XmlBeanFactory factory= new XmlBeanFactory( new ClassPathResource("com/spring/applicationcontext.xml"));
  7.  
    factory.getBean("lifeBean");
  8.  
    }
  9.  
    }

 

    运行之后,我们可以看到下面的结果:LifeCycleBean initializing...。说明bean的afterPropertiesSet方法被spring调用了。
    spring在装配完一个bean的所有合作者之后,会检查这个bean是否实现了InitializingBean接口,如果实现就调用该bean的afterPropertiesSet方法。
init-method
    Spring虽然可以通过InitializingBean完成bean初始化后对这个bean的回调,但是这种方式要求bean实现InitializingBean接口。一但bean实现了InitializingBean接口,那么这个bean的代码就和Spring耦合到一起了。通常情况下我不鼓励bean直接实现InitializingBean,可以使用Spring提供的init-method的功能来执行一个bean子定义的初始化方法。

bean实现:

  1.  
    package com.spring;
  2.  
     
  3.  
    public class LifeCycleBean{
  4.  
    publicvoid init(){
  5.  
    System. out .println("LifeCycleBean.init...");
  6.  
    }
  7.  
    }

在Spring配置文件中配置这个bean:

  1.  
    <beans>
  2.  
    <bean name ="lifeBean" class ="research.spring.beanfactory.ch4.LifeCycleBean" init-method ="init"></bean>
  3.  
    </beans>
    当spring实例化lifeBean时,你会看到控制台上打印出LifeCycleBean.init...
    Spring要求init-method是一个无参的方法,如果init-method指定的方法中有参数,那么Spring将会抛出java.lang.NoSuchMethodException。init-method指定的方法可以是public、protected以及private的,并且方法也可以是final的。另外,init-method指定的方法可以是声明为抛出异常的,就像这样:
  1.  
    final protected void init() throws Exception{
  2.  
    System.out.println("init method...");
  3.  
    if(true)
  4.  
    throw new Exception("init exception");
  5.  
    }
    如果在init-method方法中抛出了异常,那么Spring将中止这个Bean的后续处理,并且抛出一个org.springframework.beans.factory.BeanCreationException异常。
&nbsp;&nbsp;InitializingBean和init-method可以一起使用,如果一起使用时,Spring会先处理InitializingBean再处理init-method。
    org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory完成一个Bean初始化方法的调用工作。AbstractAutowireCapableBeanFactory是XmlBeanFactory的父类DefaultListableBeanFactory的父类,也就是XmlBeanFactory的超类,在AbstractAutowireCapableBeanFactory的invokeInitMethods方法中实现了一个调用Bean初始化的方法:

源代码如下:

 

 

  1.  
    //bean装配完成之后,执行bean初始化方法
  2.  
    protected void invokeInitMethods(String beanName, final Object bean, RootBeanDefinition mbd)
  3.  
    throws Throwable
  4.  
    {
  5.  
    //判断bean是否实现了InitializingBean接口
  6.  
    boolean isInitializingBean = bean instanceof InitializingBean;
  7.  
    if(isInitializingBean && (mbd == null || !mbd.isExternallyManagedInitMethod("afterPropertiesSet")))
  8.  
    {
  9.  
    if(logger.isDebugEnabled())
  10.  
    logger.debug((new StringBuilder()).append("Invoking afterPropertiesSet() on bean with name ‘").append(beanName).append("‘").toString());
  11.  
    if(System.getSecurityManager() != null)
  12.  
    try
  13.  
    {
  14.  
    AccessController.doPrivileged(new PrivilegedExceptionAction() {
  15.  
     
  16.  
    public Object run()
  17.  
    throws Exception
  18.  
    {
  19.  
    //调用afterPropertiesSet方法
  20.  
    ((InitializingBean)bean).afterPropertiesSet();
  21.  
    return null;
  22.  
    }
  23.  
     
  24.  
    final Object val$bean;
  25.  
    final AbstractAutowireCapableBeanFactory this$0;
  26.  
     
  27.  
     
  28.  
    {
  29.  
    this.this$0 = AbstractAutowireCapableBeanFactory.this;
  30.  
    bean = obj;
  31.  
    super();
  32.  
    }
  33.  
    }
  34.  
    , getAccessControlContext());
  35.  
    }
  36.  
    catch(PrivilegedActionException pae)
  37.  
    {
  38.  
    throw pae.getException();
  39.  
    }
  40.  
    else
  41.  
    ((InitializingBean)bean).afterPropertiesSet();调用afterPropertiesSet方法
  42.  
    }
  43.  
    if(mbd != null)
  44.  
    {
  45.  
    //判断bean是否定义了init-method方法
  46.  
    String initMethodName = mbd.getInitMethodName();
  47.  
    if(initMethodName != null && (!isInitializingBean || !"afterPropertiesSet".equals(initMethodName)) && !mbd.isExternallyManagedInitMethod(initMethodName))
  48.  
    //调用invokeCustomInitMethod方法来执行init-method定义的方法
  49.  
    invokeCustomInitMethod(beanName, bean, mbd);
  50.  
    }
  51.  
    }
  52.  
     
  53.  
    //执行bean定义的init-method方法
  54.  
    protected void invokeCustomInitMethod(String beanName, final Object bean, RootBeanDefinition mbd)
  55.  
    throws Throwable
  56.  
    {
  57.  
    String initMethodName = mbd.getInitMethodName();
  58.  
    //使用方法名,反射Method对象
  59.  
    final Method initMethod = mbd.isNonPublicAccessAllowed() ? BeanUtils.findMethod(bean.getClass(), initMethodName, new Class[0]) : ClassUtils.getMethodIfAvailable(bean.getClass(), initMethodName, new Class[0]);
  60.  
    if(initMethod == null)
  61.  
    {
  62.  
    if(mbd.isEnforceInitMethod())
  63.  
    throw new BeanDefinitionValidationException((new StringBuilder()).append("Couldn‘t find an init method named ‘").append(initMethodName).append("‘ on bean with name ‘").append(beanName).append("‘").toString());
  64.  
    if(logger.isDebugEnabled())
  65.  
    logger.debug((new StringBuilder()).append("No default init method named ‘").append(initMethodName).append("‘ found on bean with name ‘").append(beanName).append("‘").toString());
  66.  
    return;
  67.  
    }
  68.  
    if(logger.isDebugEnabled())
  69.  
    logger.debug((new StringBuilder()).append("Invoking init method ‘").append(initMethodName).append("‘ on bean with name ‘").append(beanName).append("‘").toString());
  70.  
    if(System.getSecurityManager() != null)
  71.  
    {
  72.  
    AccessController.doPrivileged(new PrivilegedExceptionAction() {
  73.  
     
  74.  
    public Object run()
  75.  
    throws Exception
  76.  
    {
  77.  
    ReflectionUtils.makeAccessible(initMethod);
  78.  
    return null;
  79.  
    }
  80.  
     
  81.  
    final Method val$initMethod;
  82.  
    final AbstractAutowireCapableBeanFactory this$0;
  83.  
     
  84.  
     
  85.  
    {
  86.  
    this.this$0 = AbstractAutowireCapableBeanFactory.this;
  87.  
    initMethod = method;
  88.  
    super();
  89.  
    }
  90.  
    }
  91.  
    );
  92.  
    try
  93.  
    {
  94.  
    AccessController.doPrivileged(new PrivilegedExceptionAction() {
  95.  
     
  96.  
    public Object run()
  97.  
    throws Exception
  98.  
    {
  99.  
    initMethod.invoke(bean, new Object[0]);//反射执行init-method方法
  100.  
    return null;
  101.  
    }
  102.  
     
  103.  
    final Method val$initMethod;
  104.  
    final Object val$bean;
  105.  
    final AbstractAutowireCapableBeanFactory this$0;
  106.  
     
  107.  
     
  108.  
    {
  109.  
    this.this$0 = AbstractAutowireCapableBeanFactory.this;
  110.  
    initMethod = method;
  111.  
    bean = obj;
  112.  
    super();
  113.  
    }
  114.  
    }
  115.  
    , getAccessControlContext());
  116.  
    }
  117.  
    catch(PrivilegedActionException pae)
  118.  
    {
  119.  
    InvocationTargetException ex = (InvocationTargetException)pae.getException();
  120.  
    throw ex.getTargetException();
  121.  
    }
  122.  
    } else
  123.  
    {
  124.  
    try
  125.  
    {
  126.  
    ReflectionUtils.makeAccessible(initMethod);
  127.  
    initMethod.invoke(bean, new Object[0]);//反射执行init-method方法
  128.  
    }
  129.  
    catch(InvocationTargetException ex)
  130.  
    {
  131.  
    throw ex.getTargetException();
  132.  
    }
  133.  
    }
  134.  
    }

 

    通过分析上面的源代码我们可以看到,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上。
 
可以借助这个InitializingBean方法来完成一些需要在bean初始化时完成的工作。
示例场景:
一个bean在初始化时需要读取项目目录中某个文件夹下的配置文件。
bean配置:
  1.  
    <bean id="testServiceConfig" class="com.lmb.client.TestlServiceConfigImpl">
  2.  
    <property name="folderName" value="test_config" />
  3.  
    </bean>

调用代码:

 

 

    1.  
      //bean初始化之后调用afterPropertiesSet方法根据配置目录,读取相应目录下的配置文件
    2.  
      public class TestServiceConfigImpl implements InitializingBean {
    3.  
      private String folderName;
    4.  
       
    5.  
      @Override
    6.  
      public void afterPropertiesSet()throws Exception{
    7.  
      folderName = folderName != null ? folderName : "test_config";
    8.  
      reload(folderName);
    9.  
      }
    10.  
       
    11.  
      //根据文件目录加载配置文件
    12.  
      public void reload(String fileName) throws Exception {
    13.  
       
    14.  
      fileName = fileName.startsWith("/") ? fileName.substring(1) : fileName;
    15.  
      //根据文件名称获取相应的文件
    16.  
      File file = ResourceUtils.getFile("classpath:" + fileName);
    17.  
      if (file.isDirectory()) {
    18.  
      File[] files = file.listFiles();
    19.  
      for (File configFile : files) {
    20.  
      //加载文件……
    21.  
      }
    22.  
      }
    23.  
      }
    24.  
      }

afterPropertiesSet方法和init-method区别和应用

原文:https://www.cnblogs.com/zsw1024520/p/13731548.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!