Bean的一生(Bean的生命周期)

1. 什么是Bean?

Bean是spring中组成应用程序的主体由spring IoC容器所管理的对象(IoC容器初始化、装配及管理的对象)。如果把spring比作一座大型工厂,那么bean就是该工厂的产品。

题外话:bean这个单词翻译成中文有很多意思,豆子、嘴峰、毫无价值的东西...而事实上它是spring的核心,价值非凡。这个命名可能来自于开发者的一点恶趣味。如果留心,生活处处是惊喜,你可以在编程中获得不少乐趣。比如:Git翻译过来有“饭桶、无用的人”的意思,redis的端口6379来自于MERZ(长期以来被开发者antirez及其朋友当作愚蠢的代名词)...

2. 在了解Bean的一生前,你需要了解一些东西

2.1 spring容器

  • 容器是spring框架实现的核心,负责了对象整个生命周期的管理(创建,装配,销毁)。
  • 容器是核心,而容器不唯一(spring容器 ∈ IoC容器)。并不是说只有Spring的容器才叫IOC容器,基于IOC容器的框架还有很多,并不是Spring特有的。框架本身就提供了很多个容器的实现。大概分为两种类型:一种是不常用的BeanFactory,这是最简单的容器,只能提供基本的DI功能;还有一种就是继承了BeanFactory后派生而来的应用上下文。

2.2 应用上下文ApplicationContexts

应用上下文是spring容器抽象的一种实现,一种容器对象。

  • 其抽象接口ApplicationContext的本质:一个维护Bean定义以及对象之间协作关系的高级接口。

    比起BeanFactory,它能提供更多企业级的服务,例如解析配置文本信息等等,这也是应用上下文实例对象最常见的应用场景。有了上下文对象,我们就能向容器注册需要Spring管理的对象了。对于上下文抽象接口,Spring也为我们提供了多种类型的容器实现,供我们在不同的应用场景选择。详情请看:Spring基础篇——Spring容器和应用上下文理解

2.3 Aware

spring DI最大的亮点是所有bean对spring容器的存在是没有意识的(即可替换容器,此时bean间耦合度很低),但实际开发中很有可能要用到spring本身的功能资源,这时候bean需要意识到spring容器的存在,才能调用spring所提供的资源。spring Aware中的接口可以让我们可获取spring的资源。

2.4 后置处理器BeanPostProcessor

此接口允许自定义修改新bean的一个实例,让我们可以在spring容器完成bean实例化、配置以及其他初始化方法前后添加一些自己的逻辑处理。含有两个方法,接口声明如下:

public interface BeanPostProcessor {
    //bean初始化方法调用前被调用
    Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException;
    //bean初始化方法调用后被调用
    Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException;
}

2.5 Bean的自定义初始化和销毁方法(证明代码)

如果要在bean初始化后销毁前需要执行特定的操作:

  • 使用注解,在指定的方法前加@PostConstruct / @PreDestory注解来指定该方法在初始化之后还是销毁之前调用;
  • 在xml配置bean时指定init-method / destroy-method属性指定初始化后/销毁之前调用的操作方法;
  • 实现InitializingBean / DisposableBean接口来定制初始化之后/销毁之前的操作方法
    InitializingBean:运行afterPropertiesSet()在所有bean属性被设置后
    DisposableBean:运行destory()在spring容器释放该bean之后

他们的执行顺序如下:
Bean在实例化时:Constructor → @PostConstruct → InitializingBean → init-method
Bean在销毁的过程中:@PreDestroy → DisposableBean → destroy-method

3. Bean创建对象过程

  1. 创建对象实例

    Object o = new Object();

  2. 依赖注入

    o.setXxx(...)

  3. spring bean扩展方法

    init-method, BeanPostProcessor, xxxAware等

4. Bean实例化的三种方式 详情

  • 构造器实例化
    spring容器通过bean对应的默认的构造函数来实例化bean。

  • 静态工厂方式实例化
    首先创建一个静态工厂类,在类中定义一个静态方法创建实例。

  • 实例工厂方式实例化
    该种方式的工厂类中,不再使用静态方法创建Bean实例,而是采用直接创建Bean实例的方式。同时在配置文件中,需要实例化的Bean也不是通过class属性直接指向其实例化的类,而是通过factory-bean属性配置一个实例工厂,然后使用factory-method属性确定使用工厂中哪个方法。

5. Bean实例生命周期的执行过程

  1. 初始化spring容器(其实是第0步,bean生命周期不包括,但却是bean所必须的,也许可以看作产房?);

  2. spring对bean实例化,默认是单例(Bean的作用域);

  3. 检查Aware相关接口并设置相关依赖:

    1. 如果实现了BeanNameAware接口,spring将bean的id传给setBeanName()方法;
    2. 如果实现了BeanFactoryAware接口,spring调用setBeanFactory()方法,将BeanFactory实例传进来;
    3. 如果实现了ApplicationContextAware接口,spring调用setApplicationContext()会被调用,将应用上下文的应用传入到Bean中
  4. BeanPostProcessor前置处理(如果有),调用postProcessBeforeInitialization()

  5. 初始化相关:如果实现了InitializingBean接口,spring调用afterPropertiesSet();类似地,如果实现了init-method属性声明了初始化方法,该方法也会被调用;亦或是@PostConstruct注解

  6. BeanPostProcessor后置处理(如果有),调用postProcessorAfterInitialization()

  7. 注册必要的Destruction相关回调接口

  8. 此时bean已经准备就绪,可被程序使用了,将一直驻留在应用上下文中,直到该应用上下文被销毁

  9. 销毁相关: 如果实现了DisposableBean接口,spring调用destory();类似地,还有自定义销毁方法destory-method和注解@PreDestroy

posted @ 2019-01-11 15:31  RosaDarker  阅读(...)  评论(...编辑  收藏