Spring_Spring bean 通过实现 InitializingBean ,DisposableBean 接口实现初始化方法和销毁前操作

时间:2022-03-04 19:40:43

via: http://blog.csdn.net/topwqp/article/details/8681573

 

Spring bean 通过实现 InitializingBean ,DisposableBean 接口实现初始化方法和销毁前操作

 

关于在spring  容器初始化 bean 和销毁前所做的操作定义方式有三种:

第一种:通过@PostConstruct 和 @PreDestroy 方法 实现初始化和销毁bean之前进行的操作

第二种是:通过 在xml中定义init-method 和  destory-method方法

第三种是: 通过bean实现InitializingBean和 DisposableBean接口

 

1:定义相应类实现InitializingBean ,DisposableBean 接口

import javax.annotation.PostConstruct;  

import javax.annotation.PreDestroy;  

import org.springframework.beans.factory.DisposableBean;  

import org.springframework.beans.factory.InitializingBean;  

  

public class PersonService  implements InitializingBean,DisposableBean{  

    private String  message;  

    public String getMessage() {  

        return message;  

    }  

    public void setMessage(String message) {  

        this.message = message;  

    }  

    @Override  

    public void destroy() throws Exception {  

        System.out.println("I'm  init  method  using implements DisposableBean interface...."+message);  

    }  

    @Override  

    public void afterPropertiesSet() throws Exception {  

        System.out.println("I'm  init  method  using implements InitializingBean interface...."+message);  

    }  

}  

 

2:定义相应的配置文件:

<?xml version="1.0" encoding="UTF-8"?>  

<beans xmlns="http://www.springframework.org/schema/beans"  

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  

xmlns:context="http://www.springframework.org/schema/context"  

xsi:schemaLocation="http://www.springframework.org/schema/beans  

http://www.springframework.org/schema/beans/spring-beans-3.1.xsd  

http://www.springframework.org/schema/context  

http://www.springframework.org/schema/context/spring-context-3.1.xsd">   

<bean id="personService" class="com.myapp.core.annotation.init.PersonService">  

          <property name="message" value="123"></property>  

</bean>  

</beans>

 

3:测试类:

import org.springframework.context.ApplicationContext;  

import org.springframework.context.support.AbstractApplicationContext;  

import org.springframework.context.support.ClassPathXmlApplicationContext;  

  

public class MainTest {  

    public static void main(String[] args) {  

        AbstractApplicationContext  context = new ClassPathXmlApplicationContext("resource/annotation.xml");  

        PersonService   personService  =  (PersonService)context.getBean("personService");  

context.registerShutdownHook();  

    }  

  

}  

 

 

via: http://blog.csdn.net/hikvision_java_gyh/article/details/38346827?utm_source=tuicool

spring的InitializingBean的 afterPropertiesSet 方法 和 init-method配置的 区别联系

InitializingBean

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

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

import org.springframework.beans.factory.InitializingBean;

public  class LifeCycleBean implements InitializingBean{

public  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">

</bean>

</beans>

 

编写测试程序进行测试:

package research.spring.beanfactory.ch4;

import org.springframework.beans.factory.xml.XmlBeanFactory;

import org.springframework.core.io.ClassPathResource;

public class LifeCycleTest {

public 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方法。

 

init-method

    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上。