详述 Spring 中 Bean 的生命周期

对于普通的 Java 对象,当我们使用new关键字创建对象的时候,如果它没有任何引用,则其会被垃圾回收机制回收。而由 Spring IoC 容器托管的对象,它们的生命周期则是完全由容器控制。在 Spring 中,每个 Bean 的生命周期大致如下:

bean-life

1 实例化 Bean

对于BeanFactory容器来说,当用户向容器请求一个尚未初始化的 Bean 或初始化 Bean 的时候,如果需要注入另一个尚未初始化的依赖,容器就会调用createBean进行实例化;对于ApplicationContext容器来说,当容器启动结束后,便实例化所有的 Bean。

容器通过获取BeanDefinition对象中的信息进行实例化。并且这一步仅仅是简单的实例化,并未进行依赖注入。 实例化对象被包装在BeanWrapper对象中,BeanWrapper提供了设置对象属性的接口,从而避免了使用反射机制设置属性。

2 设置对象属性(依赖注入)

实例化后的对象被封装在BeanWrapper对象中,并且此时对象仍然是一个原生的状态,并没有进行依赖注入。 紧接着,Spring 根据BeanDefinition中的信息进行依赖注入,并且通过BeanWrapper提供的设置属性的接口完成依赖注入。

3 注入 Aware 接口

紧接着,Spring 会检测该对象是否实现了xxxAware接口,并将相关的xxxAware实例注入给 Bean:

  • 如果 Bean 实现了BeanNameAware接口,Spring 将 Bean 的 ID 传递给setBeanName()方法。实现BeanNameAware主要是为了通过 Bean 的引用来获得 Bean 的 ID,一般业务中是很少有用到 Bean 的 ID 的。
  • 如果 Bean 实现了BeanFactoryAware接口,Spring 将调用setBeanDactory(BeanFactory bf)方法并把BeanFactory容器实例作为参数传入。实现BeanFactoryAware主要目的是为了获取 Spring 容器,如 Bean 通过 Spring 容器发布事件等。
  • 如果 Bean 实现了ApplicationContextAware接口,Spring 容器将调用setApplicationContext(ApplicationContext ctx)方法,把应用上下文作为参数传入,作用与BeanFactory类似都是为了获取 Spring 容器,不同的是 Spring 容器在调用setApplicationContext方法时会把它自己作为setApplicationContext的参数传入,而 Spring 容器在调用setBeanDactory前需要程序员自己指定(注入)setBeanDactory里的参数BeanFactory

4 BeanPostProcessor

当经过上述几个步骤后,Bean 对象已经被正确构造,但如果你想要对象被使用前再进行一些自定义的处理,就可以通过BeanPostProcessor接口实现。 该接口提供了两个函数:

  • postProcessBeforeInitialzation(Object bean, String beanName)当前正在初始化的 Bean 对象会被传递进来,我们就可以对这个 Bean 作任何处理。这个函数会先于InitialzationBean执行,因此称为前置处理。 所有Aware接口的注入就是在这一步完成的。
  • postProcessAfterInitialzation(Object bean, String beanName)当前正在初始化的 Bean 对象会被传递进来,我们就可以对这个 Bean 作任何处理。这个函数会在InitialzationBean完成后执行,因此称为后置处理。

5 InitializingBean 与 init-method

BeanPostProcessor的前置处理完成后就会进入本阶段。 InitializingBean接口只有一个函数:

  • afterPropertiesSet()

这一阶段也可以在 Bean 正式构造完成前增加我们自定义的逻辑,但它与前置处理不同,由于该函数并不会把当前 Bean 对象传进来,因此在这一步没办法处理对象本身,只能增加一些额外的逻辑。 若要使用它,我们需要让 Bean 实现该接口,并把要增加的逻辑写在该函数中。然后,Spring 会在前置处理完成后检测当前 Bean 是否实现了该接口,并执行afterPropertiesSet函数。

当然,Spring 为了降低对客户代码的侵入性,给 Bean 的配置提供了init-method属性,该属性指定了在这一阶段需要执行的函数名。Spring 便会在初始化阶段执行我们设置的函数。init-method本质上仍然使用了InitializingBean接口。

6 DisposableBean 和 destroy-method

如果 Bean 实现了DispostbleBean接口,Spring 将调用它的destory方法,作用与在配置文件中对 Bean 使用destory-method属性的作用一样,都是在 Bean 实例销毁前执行的方法。

至此,Spring 中 Bean 的经历了从创建到消耗的整个生命周期的过程。


参考资料:本文的内容借鉴了「知乎」中「Spring中Bean的生命周期是怎样的?」这篇文章的回答。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

安正勋

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值