spring容器通过反射的方式创建bean对象,在创建之前需要知道bean对象的类信息,定义bean对象有如下2种方式:
定义bean对象形式多种多样
xml、注解、properties、json、yaml
例如xml:
首先加载xml文件到内存,得到io流,使用sax或dom4j将xml解析成document对象,然后得到父子节点node,然后设置到BeanDefinition中
不同的定义方式对应不同的解析方式,而spring容器需要一个统一的规范来接收解析好的bean信息。
spring容器通过BeanDefinition来接收定义好的bean信息。
所有需要进行解析的操作都需要实现BeanDefinitionReader接口。
spring容器通过BeanDefinition根据bean定义好的类信息通过反射的方式创建对象、使用和销毁。
为了完成某些具体的扩展功能,有以下两种后置处理器的实现方式:
- • BeanFactoryPostProcessor
该处理器生成BeanFactory
- • BeanPostProcessor
该处理器生成Bean
而要使用spring容器中的bean对象必须要通过BeanFactory。
我们经常使用ApplictionContext中的getBean获取bean对象
context.getBean(Person.class)
Bean 工厂实现应支持标准 Bean 生命周期接口, 上图为bean完整的初始化方法集及其顺序。
beanFactory怎么实现的扩展呢
举例说明:
图[xml定义bean]中${jdbc.username},在BeanDefined读取到的时候,属性值并没有改变,什么时候才真正替换为真实值呢?
这里有一个很重要的接口BeanFactoryPostProcessor
该接口的一个实现类PlaceholderConfigurerSupport,该类的作用是解析在 Bean 定义属性值中占位符的属性资源配置器的抽象基类。
获取BeanDefined中property中的value值然后解析占位符${jdbc.username},替换为真实值。
类似这样的扩展可以实现很多,预留出很多的扩展点,在需要增强的时候任意扩展,这就是BeanFactoryPostProcessor类的核心思想所在。
注解
解析xml生成bean的流程是原有的标准解析流程,而解析注解的方式生成bean是在原有保准流程之上做的扩展。
BeanFactoryPostProcessor的子类ConfigurationClassPostProcessor,这个类用来解析各种注解。
创建对象的过程
创建对象包含两个环节,实例化和初始化。
实例化只是在内存中开辟空间,初始化是属性赋值。
具体的bean对象是保存在spring容器中map中
- • singletonObjects 一级缓存对象集合
- • earlySingletonObjects 二级缓存对象集合
- • singletonFactories 三级缓存对象集合