(spring-第6回【IoC基礎篇】)BeanDefinition——實例化Bean之前的第一大利器。


上節講了Bean實例化的內部機制,這里再復述一遍:

  1. ResourceLoader從系統中加載XML配置信息,並由Resource來表示。
  2. BeanDefinitionReader從Resource中讀取配置信息,把配置文件中的<bean>解析成一個BeanDefinition對象,然后把BeanDefinition對象放到BeanDefinitionRegistry注冊表中。
  3. 容器從BeanDefinitionRegistry注冊表中掃描出Bean工廠后處理器的Bean(該Bean實現了BeanFactoryPostProcessor),用這個工廠后處理器來加工BeanDefinitionRegistry注冊表中的所有BeanDefinition對象。具體做了兩件事:
    1.   對使用到<bean>元素的占位符的Bean進行解析,把占位符轉換成具體值,從而把半成品的BeanDefinition對象轉為成品的對象。
    2.   掃描BeanDefinitionRegistry注冊表中的所有BeanDefinition對象,通過java反射機制找出所有屬性編輯器的Bean(實現了PropertyEditor的Bean),然后把它放到屬性編輯器注冊表中(PropertyEditorRegistry)。
  4. 容器從BeanDefinitionRegistry中取出加工過的BeanDefinition,並調用InstantiationStrategy着手bean的實例化工作。
  5. 在實例化Bean時,Spring容器使用BeanWrapper對Bean進行封裝,BeanWrapper結合BeanDefinition以及屬性編輯器完成Bean屬性的設置工作。
  6. 利用容器中注冊的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部分之后,我將對源碼進行剖析,深入探究其奧妙,希望大家繼續支持並關注我的博客。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM