spring完成xml配置的解析之后,解析的document的结果存放在DeferredDocumentImpl,接下来呢,就是spring根据这个document进行bean的注册、实例化,也就是我们常说的IOC、Aop了。本文档先主要说明IOC。
我们从XmlBeanDefinitionReader.registerBeanDefinitions()开始说起
XmlBeanDefinitionReader.registerBeanDefinitions() --》
DefaultBeanDefinitionDocumentReader 把document转换为bean --》
DefaultBeanDefinitionDocumentReader.doRegisterBeanDefinitions --》
DefaultBeanDefinitionDocumentReader
.parseBeanDefinitions
到这里之后呢,分两种情况进行document转换bean的定义:一种是spring自带的标签,比如bean、import、alias、beans,采用spring自带的解析,如processBeanDefinition方法;另一种就是自定义的delegate.parseCustomElement(ele)。
先介绍第一种spring自带的标签解析方法。
ApplicationContext的xml解析及bean的注册在AbstractApplicationContext的refresh()中已经分的比较清晰的了,每个阶段对应不同的方法。
根据重要性,我把spring的bean的注册分为几个阶段:
第一阶段,是以AbstractApplicationContext.obtainFreshBeanFactory()方法入口,解析xml完毕之后,进入spring默认的bean注册方法processBeanDefinition(),这一阶段应该是比较简单,就是对document进行遍历(以bean标签说明),把bean的id、beanDefinition存储到beanDefinitionMap里面。BeanDefinition对应spring容器中的bean的定义,就是把xml中配置的内容解析到BeanDefinition。
第二阶段,就是添加前置、后置处理器,比如在prepareBeanFactory添加“ApplicationContextAwareProcessor”处理器,在registerBeanPostProcessors添加“BeanPostProcessorChecker、AnnotationAwareAspectJAutoProxyCreator、ApplicationListenerDetector”处理器。这部分到时在介绍aop时着重讲解。
第三阶段,bean的实例化,以finishBeanFactoryInitialization()为入口,对doGetBean(),getBean(),getObject(),createBean()等方法调用,对bean进行实例化,首先初始化实例时,实例的属性并没有赋予xml中配置的属性值。
在AbstractAutowireCapableBeanFactory的populateBean方法中进行属性的赋值。BeanWrapperImpl包装bean之后,一般在BeanDefinitionValueResolver
.resolveValueIfNecessary
BeanWrapperImpl.converIfNecessary()--->TypeConverterDelegate.converIfNecessary()…等方法已经完成了转换及赋值,特殊的可能需要自定义,需要进行类型转换。
resolveValueIfNecessary()呢,是根据类型把返回相应的对象值,比如string、list、array等,需要经过处理。 converIfNecessary,这块跟那个内省PropertyDescriptor有点类似,通过属性名称获取类型,set值。通过内省PropertyDescriptor传入Class,获取到该类的所有的属性、所有的读写方法等,通过属性的类型到注册了类型转换的map中进行查找,通过类型转换器setAsText把值转正确,然后通过属性的写方法set方法进行值的赋予。
PropertyEditorRegistrySupport中注册了很多定义的数据类型。
BeanWrapperImpl对BeanDefinition进行了包装、进行了类的实例化和属性赋值。
定义bead的时候,最重要的其实就是给BeanDefinition设置Class。
指针(对象)的传递:
delegate--》readerContext--》reader--》registry
更进一步说明,最后的bean存放结果如下:DefaultListableBeanFactory的beanDefinitionMap,beanDefinitionMap在DefaultBeanDefinitionDocumentReader的readerContext的reader(XMLBeanDefinitionReader)的registry中。
delegate是DefaultBeanDefinitionDocumentReader中的成员属性
registry对应DefaultListableBeanFactory。
自定义标签的开发步骤通过书本及网络可以得到,基本上步骤是固定的,按照步骤编码。通过对比自定义标签解析,会收货更大。我会用另外一篇文章编写。
以Aware结尾的类,spring会把spring容器设置到该类里面(setBeanFactory,
setApplicationContext)
ignoreDependencyInterface() 忽略自动装配(autowring),对应配置文件beans元素的default-autowrie。与@autowired是两回事。
XmlBeanFactory中直接使用spring的xml配置文件的文件流时,会报错:
new XMLBeanFactory(new
InputStreamResource(Main.class.getResourceAsStream(“beans.xml”))),会报错文件打开了,在XMLBeanDefinitionReader的detectValidationMode中,resouce.isOpen()中判断的。文件已经被打开所以报错。
可以使用相对路径: ClassPathResouce(“xx/bb/beans.xml”)
相比<context:annotation-config>, <context:component-scan>功能跟多,可以扫描指定的包,对没注册的bean实例化。
BeanFactory与ApplicationContext相比,beanFactory只提供了实例化、存储bean的功能,而ApplicationContext则除了BeanFactory的所有功能之后,还提供了aop,消息、事务管理等。 BeanFactory是没有aop功能的。
BeanFactory 与 FactoryBean相比,BeanFactory是spring的bean容器, FactoryBean是通过Java去简化spring的配置。