上節講了Bean實例化的內部機制,這里再復述一遍:
- ResourceLoader從系統中加載XML配置信息,並由Resource來表示。
- BeanDefinitionReader從Resource中讀取配置信息,把配置文件中的<bean>解析成一個BeanDefinition對象,然后把BeanDefinition對象放到BeanDefinitionRegistry注冊表中。
- 容器從BeanDefinitionRegistry注冊表中掃描出Bean工廠后處理器的Bean(該Bean實現了BeanFactoryPostProcessor),用這個工廠后處理器來加工BeanDefinitionRegistry注冊表中的所有BeanDefinition對象。具體做了兩件事:
- 對使用到<bean>元素的占位符的Bean進行解析,把占位符轉換成具體值,從而把半成品的BeanDefinition對象轉為成品的對象。
- 掃描BeanDefinitionRegistry注冊表中的所有BeanDefinition對象,通過java反射機制找出所有屬性編輯器的Bean(實現了PropertyEditor的Bean),然后把它放到屬性編輯器注冊表中(PropertyEditorRegistry)。
- 容器從BeanDefinitionRegistry中取出加工過的BeanDefinition,並調用InstantiationStrategy着手bean的實例化工作。
- 在實例化Bean時,Spring容器使用BeanWrapper對Bean進行封裝,BeanWrapper結合BeanDefinition以及屬性編輯器完成Bean屬性的設置工作。
- 利用容器中注冊的Bean后處理器(該Bean實現了BeanPostProcessor)對第五步生成的Bean進行后續加工。
從實例化的過程中可以看出,BeanDefinition起到中流砥柱的作用。因為BeanDefinition是配置文件<bean>元素標簽在容器中的內部表示。比如,<bean>標簽在XML中有class,scope,lazy-init等屬性,那么在BeanDefinition中則有相應的beanClass,scope,lazyInit屬性等。
BeanDefinition接口的繼承結構如圖:
頂級的BeanDefinition其實是個接口,下面的AbstractBeanDefinition實現了這個接口,而最下面的ChildBeanDefinition和RootBeanDefinition分別繼承了AbstractBeanDefinition。來看一個XML配置:
1 <!-- 父子<bean> --> 2 <bean id="car0" class="com.baobaotao.tagdepend.Car" 3 p:brand="紅旗CA72" p:price="2000.00" p:color="黑色" 4 abstract="true"/> 5 6 <bean id="car3" parent="abstractCar"> 7 <property name="color" value="紅色"/> 8 </bean>
car3繼承了car0,對應繼承結構圖,car3使用的是ChildBeanDefinition,car0使用的是RootBeanDefinition,也就是說,car3的屬性多於car0,而不管是car3還是car0,它們都有共性(即使配置文件中沒寫,也是隱形存在的,比如上面講的scope,lazyInit等),這些共性將提取在AbstractBeanDefinition中。如果<bean>標簽沒有繼承關系,那么它將使用默認的RootBeanDefinition,在2.5版本之后加了一個GenericBeanDefinition,因為自身優勢而成為默認的使用類。
下面是BeanDefinition的API:(在線文檔:http://tool.oschina.net/apidocs/apidoc?api=Spring-3.1.1)
在方法概述中,我們可以看到這個接口中定義了所有<bean>屬性的方法接口,比如singleton、prototype,lazyInit等。
下面是AbstractBeanDefinition的結構圖和API:
由於API里的方法很多,我截取幾個例子:
這些get方法是獲取共性類的屬性值,實際上,共性屬性的默認值在定義成員變量時就已經默認給定了,請看代碼:
1 private volatile Object beanClass; 2 3 private String scope = SCOPE_DEFAULT; 4 5 private boolean singleton = true; 6 7 private boolean prototype = false; 8 9 private boolean abstractFlag = false; 10 11 private boolean lazyInit = false; 12 13 private int autowireMode = AUTOWIRE_NO; 14 15 private int dependencyCheck = DEPENDENCY_CHECK_NONE; 16 17 private String[] dependsOn; 18 19 private boolean autowireCandidate = true; 20 21 private boolean primary = false;
上面代碼截取的是AbstractBeanDefinition的源碼,可以看到許多<bean>標簽的屬性默認值。
一開始創建的BeanDefinition由於占位符的原因是個半成品,需要用Bean工廠后處理器對Bean進行處理,處理完之后由InstantiationStrategy對BeanDefinition進行實例化。后面我們會詳細介紹InstantiationStrategy。
BeanDefinition是如何創建的,XML又是如何被解析的?這些問題需要深入到源碼中去尋求答案,在講完IoC部分之后,我將對源碼進行剖析,深入探究其奧妙,希望大家繼續支持並關注我的博客。