Spring之bean的生命周期

Spring作为当前Java最流行、最强大的轻量级框架,受到了程序员的热烈欢迎。准确的了解Spring Bean的生命周期是非常必要的。我们通常使用ApplicationContext作为Spring容器。这里,我们讲的也是 ApplicationContext中Bean的生命周期。而实际上BeanFactory也是差不多的,只不过处理器需要手动注册。
一、生命周期流程图:
  Spring Bean的完整生命周期从创建Spring容器开始,直到最终Spring容器销毁Bean,这其中包含了一系列关键点。

Spring中Bean的生命周期

若容器注册了以上各种接口,程序那么将会按照以上的流程进行。下面将仔细讲解各接口作用。

二、各种接口方法分类
  Bean的完整生命周期经历了各种方法调用,这些方法可以划分为以下几类:
1.Bean自身的方法:这个包括了Bean本身调用的方法和通过配置文件中<bean>的init-methoddestroy-method指定的方法
2.Bean及生命周期接口方法:这个包括了BeanNameAwareBeanFactoryAwareInitializingBeanDiposableBean这些接口的方法
3.容器级生命周期接口方法:这个包括了InstantiationAwareBeanPostProcessorBeanPostProcessor 这两个接口实现,一般称它们的实现类为“后处理器”。
4.工厂后处理器接口方法:这个包括了AspectJWeavingEnabler, ConfigurationClassPostProcessor, CustomAutowireConfigurer等等非常有用的工厂后处理器接口的方法。工厂后处理器也是容器级的。在应用上下文装配配置文件之后立即调用。
  
三、演示
  我们用一个简单的Spring Bean来演示一下Spring Bean的生命周期。
1、首先是一个简单的Spring Bean,调用Bean自身的方法和Bean级生命周期接口方法,为了方便演示,它实现了BeanNameAware、BeanFactoryAware、InitializingBean和DiposableBean这4个接口,同时有2个方法,对应配置文件中<bean>的init-method和destroy-method。如下:

package com.kong.sitemush.bean;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.*;

/**
 * Created by Kong on 2016/9/22.
 */
public class User implements BeanFactoryAware,BeanNameAware,InitializingBean,DisposableBean {

    private Long id;

    private String name;

    private String address;

    private BeanFactory beanFactory;

    private String beanName;

    public User(){
        System.out.println("创建了User 对象!");
    }
    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        System.out.println("注入了User name!");
        this.name = name;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    // 这是BeanFactoryAware接口方法
    @Override
    public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
        System.out.println("【BeanFactoryAware接口】调用BeanFactoryAware.setBeanFactory()");
        this.beanFactory = beanFactory;
    }

    @Override
    public void setBeanName(String name) {
        System.out.println("调用【BeanNameAware】的setBeanName方法");
    }

    @Override
    public void destroy() throws Exception {
        System.out.println("调用【DisposableBean】的destroy方法!");
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        System.out.println("调用【InitializingBean】的afterPropertiesSet方法");
    }
}

2、接下来是演示BeanPostProcessor接口的方法,如下:

package com.kong.sitemush.bean;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;

/**
 * Created by Administrator on 2016/9/22.
 */
public class MyBeanPostProcessor implements BeanPostProcessor {



    public MyBeanPostProcessor(){
        super();
        System.out.println("调用MyBeanPostProcessor的构造方法!");
    }
    //初始化Bean之前调用
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("调用BeanPostProcessor接口方法postProcessAfterInitialization对属性进行更改!");
        return bean;
    }
    //初始化Bean之后调用
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("调用BeanPostProcessor接口的方法postProcessAfterInitialization对属性进行修改");
        return bean;
    }
}

如上,BeanPostProcessor接口包括2个方法postProcessAfterInitialization和postProcessBeforeInitialization,这两个方法的第一个参数都是要处理的Bean对象,第二个参数都是Bean的name。返回值也都是要处理的Bean对象。这里要注意。

3、InstantiationAwareBeanPostProcessor 接口本质是BeanPostProcessor的子接口,一般我们继承Spring为其提供的适配器类InstantiationAwareBeanPostProcessor Adapter来使用它,如下:

package com.kong.sitemush.bean;

import org.springframework.beans.BeansException;
import org.springframework.beans.PropertyValues;
import org.springframework.beans.factory.config.InstantiationAwareBeanPostProcessorAdapter;

import java.beans.PropertyDescriptor;

/**
 * Created by Administrator on 2016/9/22.
 *  public class MyInstantiationAwareBeanPostProcessor extends
 10         InstantiationAwareBeanPostProcessorAdapter {
 */
public class MyInstantiationAwareBeanPostProcessor extends InstantiationAwareBeanPostProcessorAdapter{

    public MyInstantiationAwareBeanPostProcessor(){
        super();
        System.out.println("这是InstantiationAwareBeanPostProcessorAdapter的构造器!!");
    }

    // 接口方法、实例化Bean之前调用
    @Override
    public Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) throws BeansException {
        System.out.println("InstantiationAwareBeanPostProcessor调用postProcessBeforeInstantiation方法");
        return super.postProcessBeforeInstantiation(beanClass,beanName);
    }



    //接口方法Bean实现之后调用
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("InstantiationAwarePostProcessor调用postProcessAfterInitialization方法");
        return super.postProcessAfterInitialization(bean, beanName);
    }


    // 接口方法、设置某个属性时调用
    @Override
    public PropertyValues postProcessPropertyValues(PropertyValues pvs, PropertyDescriptor[] pds, Object bean, String beanName) throws BeansException {
        System.out.println("调用InstantiationAwarePostProcessor的postProcessPropertyValues方法");
        return super.postProcessPropertyValues(pvs, pds, bean, beanName);
    }
}

这个有3个方法,其中第二个方法postProcessAfterInitialization就是重写了BeanPostProcessor的方法。第三个方法postProcessPropertyValues用来操作属性,返回值也应该是PropertyValues对象。

4、演示工厂后处理器接口方法,如下:

package com.kong.sitemush.bean;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;

/**
 * Created by Administrator on 2016/9/22.
 */
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {

    public MyBeanFactoryPostProcessor(){
        super();
        System.out.println("MyBeanFactoryPostProcessor的构造方法");
    }

   /**
    这时候所有的Bean定义已经被加载,但是没有进行实例化
    修改应用程序上下文的内部bean工厂标准后初始化。所有bean定义将被加载,但没有实例化bean将。
    这允许覆盖或添加属性甚至eager-initializing
    */
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        System.out.println("调用BeanFactoryPostProcessor 的postProcessBeanFactory方法");
        BeanDefinition bd = beanFactory.getBeanDefinition("user");
        bd.getPropertyValues().addPropertyValue("address", "黑龙江哈尔滨松北区");
    }
}

5、配置文件如下beans.xml,很简单,使用ApplicationContext,处理器不用手动注册:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans-4.0.xsd"
       default-lazy-init="true">

    <bean id="user" class="com.kong.sitemush.bean.User">
        <property name="name" value="郭之源"/>
    </bean>

    <bean id="myBeanFactoryPostProcessor" class="com.kong.sitemush.bean.MyBeanFactoryPostProcessor"/>

    <bean id="myInstantiationAwareBeanPostProcessor" class="com.kong.sitemush.bean.MyInstantiationAwareBeanPostProcessor"/>

    <bean id="myBeanPostProcessor" class="com.kong.sitemush.bean.MyBeanPostProcessor"/>
</beans>

6、下面测试一下:

package kong.java;


import com.kong.sitemush.bean.User;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

/**
 * Created by Administrator on 2016/9/22.
 */
public class SpringBeanLifecycleTest {


    @Test
    public void testLifeCycle(){
        System.out.println("现在开始初始化容器");

        ApplicationContext factory = new ClassPathXmlApplicationContext("applicationContext2.xml");
        System.out.println("容器初始化成功");
        //得到Preson,并使用
        User user = factory.getBean("user",User.class);
        System.out.println(user);

        System.out.println("现在开始关闭容器!");
        ((ClassPathXmlApplicationContext)factory).registerShutdownHook();
    }
}

关闭容器使用的是实际是AbstractApplicationContext的钩子方法。

我们来看一下结果:

现在开始初始化容器
MyBeanFactoryPostProcessor的构造方法
调用BeanFactoryPostProcessor 的postProcessBeanFactory方法
这是InstantiationAwareBeanPostProcessorAdapter的构造器!!
调用MyBeanPostProcessor的构造方法!
容器初始化成功
InstantiationAwareBeanPostProcessor调用postProcessBeforeInstantiation方法
创建了User 对象!
调用InstantiationAwarePostProcessor的postProcessPropertyValues方法
注入了User name!
调用【BeanNameAware】的setBeanName方法
【BeanFactoryAware接口】调用BeanFactoryAware.setBeanFactory()
调用BeanPostProcessor接口方法postProcessAfterInitialization对属性进行更改!
调用【InitializingBean】的afterPropertiesSet方法
InstantiationAwarePostProcessor调用postProcessAfterInitialization方法
调用BeanPostProcessor接口的方法postProcessAfterInitialization对属性进行修改
com.kong.sitemush.bean.User@7f3399d7
现在开始关闭容器!
调用【DisposableBean】的destroy方法!

Process finished with exit code 0
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容