三、spring之IOC

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的配置。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容