Spring ConfigurationClassPostProcessor Bean解析及自注冊過程


一、Bean的自注冊過程

  

二、自注冊過程說明

ConfigurationClassParser解析流程

   1、處理@PropertySources注解,配置信息的解析

  2、處理@ComponentScan注解:使用ComponentScanAnnotationParser掃描basePackage下的需要解析的類(@SpringBootApplication注解也包括了@ComponentScan注解,只不過basePackages是空的,空的話會去獲取當前@Configuration修飾的類所在的包[這個會在下面詳細解釋]),並注冊到BeanFactory中(這個時候bean並沒有進行實例化,而是進行了注冊。具體的實例化在finishBeanFactoryInitialization方法中執行)。對於掃描出來的類,遞歸解析

  3、處理@Import注解:先遞歸找出所有的注解,然后再過濾出只有@Import注解的類,得到@Import注解的值。比如查找@SpringBootApplication注解的@Import注解數據的話,首先發現@SpringBootApplication不是一個@Import注解,然后遞歸調用修飾了@SpringBootApplication的注解,發現有個@EnableAutoConfiguration注解,再次遞歸發現被@Import(EnableAutoConfigurationImportSelector.class)修飾,還有@AutoConfigurationPackage注解修飾,再次遞歸@AutoConfigurationPackage注解,發現被@Import(AutoConfigurationPackages.Registrar.class)注解修飾,所以@SpringBootApplication注解對應的@Import注解有2個,分別是@Import(AutoConfigurationPackages.Registrar.class)和@Import(EnableAutoConfigurationImportSelector.class)。找出所有的@Import注解之后,開始處理邏輯:

     (1)、遍歷這些@Import注解內部的屬性類集合

     (2)、如果這個類是個ImportSelector接口的實現類,實例化這個ImportSelector,如果這個類也是DeferredImportSelector接口的實現類,那么加入ConfigurationClassParser的deferredImportSelectors屬性中讓第6步處理。否則調用ImportSelector的selectImports方法得到需要Import的類,然后對這些類遞歸做@Import注解的處理

     (3)、如果這個類是ImportBeanDefinitionRegistrar接口的實現類,設置到配置類ConfigurationClass的importBeanDefinitionRegistrars屬性中

     (4)、其它情況下把這個類入隊到ConfigurationClassParser的importStack(隊列)屬性中,然后把這個類當成是@Configuration注解修飾的類遞歸重頭開始解析這個類

  4、處理@ImportResource注解:獲取@ImportResource注解的locations屬性,得到資源文件的地址信息。然后遍歷這些資源文件並把它們添加到配置類的importedResources屬性中

  5、處理@Bean注解:獲取被@Bean注解修飾的方法,然后添加到配置類的beanMethods屬性中

  6、處理DeferredImportSelector:處理第3步@Import注解產生的DeferredImportSelector,進行selectImports方法的調用找出需要import的類,然后再調用第3步相同的處理邏輯處理

@SpringBootApplication注解

  @SpringBootApplication注解被@EnableAutoConfiguration修飾,@EnableAutoConfiguration注解被@Import(EnableAutoConfigurationImportSelector.class)修飾,所以在第3步會找出這個@Import修飾的類EnableAutoConfigurationImportSelector,這個類剛好實現了DeferredImportSelector接口,接着就會在第6步被執行。第6步selectImport得到的類就是自動化配置類。

  EnableAutoConfigurationImportSelector的selectImport方法會在spring.factories文件中找出key為EnableAutoConfiguration對應的值【這些值就是所謂的自動化配置類(XXXAutoConfiguration)】。

  ConfigurationClassParser解析完成之后,被解析出來的類會放到configurationClasses屬性中。然后使用ConfigurationClassBeanDefinitionReader去解析這些類。

ComponentScanAnnotationParser包掃描相關

  首先啟動一個springboot web工程,找到ApplicationContext的父類org.springframework.context.support.AbstractApplicationContext

  

  我們發現,其真實的對象的類型是AnnotationConfigEmbeddedWebApplicationContext。 自身構造過程如下。

  

  繼續追蹤AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry)

  

  

  終於找到了ConfigurationClassPostProcessor自注冊后置處理器。這里有個有趣的ConfigurationClassPostProcessor解析流程, 內部調用中會使用 ComponentScanAnnotationParser去掃描,ComponentScanAnnotationParser類的parse方法有這樣一段邏輯

if (basePackages.isEmpty()) { basePackages.add(ClassUtils.getPackageName(declaringClass)); }

  參考下面的相關圖:

  也就是如果basePackages沒有配置,會找declaringClass 對應包及其子包,declaringClass對應springboot項目中我們寫的Application.java(@SpringBootApplication)

  

  

  

  

調用棧

   

實際項目結構

  

  包結構這樣的話,即使不配置componentScan,不使用AutoConfiguration,也可以掃描到bean。Application.java帶有@SpringbootApplication注解,該注解中帶有@ComponentScan注解,Application.java包的范圍包含了所有java文件,所以項目中所有bean都可以被掃描到。

  注:declaringClass 指的是帶有@ComponentScan 注解的類

   歡迎訪問微信訂閱號原文:ConfigurationClassPostProcessor Bean解析及自注冊過程

  就先分享這么多了,更多分享請關注我們的技術公眾吧!!!


免責聲明!

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



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