Bean的生命周期
当一个对象被Spring所管理,那么它的生命周期就完全由容器所控制。Bean的生命周期从大的方面来分可以分为以下四个阶段:
1. 实例化
2. 属性赋值
3. 初始化
4. 销毁
这样的划分是比较笼统的,接下来就从细分的角度来的Bean的生命周期做一个详细的阐述。
我们知道在Java中万物皆对象,而在Spring中被Spring管理的对象被称为bean。那么普通的对象是如何变成bean的呢?
1. 生成BeanDefinition
首先,Spring会通过BeanDefinitionReader 从XML、注解、配置类或者是其他方式,读取Bean的定义信息,来生成一个BeanDefinition,BeanDefinition接口是Spring对Bean的抽象,是用来描述Bean的。其中包含了如:bean名称、是否是单例的、是否懒加载等描述信息。
2. 推断构造方法
有了BeanDefinition之后就可以准备生成实例化的对象了,但在这之前还有一步。就是一个对象它可能会有多个构造方法,那Spring如何知道你要生成哪一个构造方法的对象呢?所以,Spring需要推断构造方法。
3. 实例化Bean
有了构造方法之后,就可以实例化对象了。当用户向容器请求一个尚未初始化的bean时,或初始化bean的时候需要注入另一个尚未初始化的依赖时,容器就会调用createBean进行实例化。注意:此处生成的对象都是空对象,也就是说它的属性都是空的,并没有进行依赖注入。
4. 依赖注入
在这一阶段,Spring就会给实例化出来的对象进行赋值。所以就要用到第一步生成的BeanDefinition,Spring根据BeanDefinition中的信息进行依赖注入。 并且通过BeanWrapper提供的设置属性的接口完成依赖注入。 而在这一步也涉及到了循环依赖这一问题。
5. 注入Aware类型接口
Aware类型的接口的作用就是让我们能够拿到Spring容器中的一些资源。如果一个Bean实现了该接口,那么当该Bean被spring初始化时,spring会向该Bean注入相关资源(就是会回调接口中的方法)。以下是对Aware接口作用的一个示例:
@Component
public class UserService implements BeanNameAware {
@Override
public void setBeanName(String s) {
System.out.println("bean的名字为:" + s);
}
}
@Test
public void test() {
ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
}
可以看到,当Spring容器初始化,UserService被初始化,也就调用了UserService所实现的BeanNameAware接口的setBeanName方法。在Spring中这样的Aware类型的接口还有很多,以下是常用的Aware相关接口作用说明:
image.png
5. 初始化
经过以上的几个步骤,bean对象就已经被正确构造了。但是如果你想要对象被使用之前,在进行一些自定义的处理,就可以定制一些初始化方法。初始化方法调用的流程是这样的:
image.png
6. 销毁
经过以上的工作后,Bean将一直驻留在应用上下文中给应用使用,直到应用上下文被销毁。Bean的销毁有两个方法:执行DisposableBean的destroy方法、执行当前对象定义的destroy方法。这两个方法的作用是相同的,都是在Bean实例销毁前执行的方法。
至此,SpringBean的生命周期就结束了。总的来说,Bean的生命周期大的方面可以分为:
实例化 ---- 属性赋值 ----- 初始化 ------ 销毁
细分可以如下图所示: