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创建对象过程
-
创建对象实例
Object o = new Object();
-
依赖注入
o.setXxx(...)
-
spring bean扩展方法
init-method, BeanPostProcessor, xxxAware等
4. Bean实例化的三种方式 详情
-
构造器实例化
spring容器通过bean对应的默认的构造函数来实例化bean。 -
静态工厂方式实例化
首先创建一个静态工厂类,在类中定义一个静态方法创建实例。 -
实例工厂方式实例化
该种方式的工厂类中,不再使用静态方法创建Bean实例,而是采用直接创建Bean实例的方式。同时在配置文件中,需要实例化的Bean也不是通过class属性直接指向其实例化的类,而是通过factory-bean属性配置一个实例工厂,然后使用factory-method属性确定使用工厂中哪个方法。
5. Bean实例生命周期的执行过程
-
初始化spring容器(其实是第0步,bean生命周期不包括,但却是bean所必须的,也许可以看作产房?);
-
spring对bean实例化,默认是单例(Bean的作用域);
-
检查Aware相关接口并设置相关依赖:
- 如果实现了BeanNameAware接口,spring将bean的id传给setBeanName()方法;
- 如果实现了BeanFactoryAware接口,spring调用setBeanFactory()方法,将BeanFactory实例传进来;
- 如果实现了ApplicationContextAware接口,spring调用setApplicationContext()会被调用,将应用上下文的应用传入到Bean中
-
BeanPostProcessor前置处理(如果有),调用postProcessBeforeInitialization()
-
初始化相关:如果实现了InitializingBean接口,spring调用afterPropertiesSet();类似地,如果实现了init-method属性声明了初始化方法,该方法也会被调用;亦或是@PostConstruct注解
-
BeanPostProcessor后置处理(如果有),调用postProcessorAfterInitialization()
-
注册必要的Destruction相关回调接口
-
此时bean已经准备就绪,可被程序使用了,将一直驻留在应用上下文中,直到该应用上下文被销毁
-
销毁相关: 如果实现了DisposableBean接口,spring调用destory();类似地,还有自定义销毁方法destory-method和注解@PreDestroy