11張流程圖搞定 Spring Bean 生命週期

專注於Java領域優質技術,歡迎關注

整理:

搜雲庫技術團隊

在網上已經有跟多Bean的生命週期的博客,但是很多都是基於比較老的版本了,最近把整個流程化成了一個流程圖。待會兒使用流程圖,說明以及代碼的形式來說明整個聲明週期的流程。注意因為代碼比較多,這裡的流程圖只畫出了大概的流程,具體的可以深入代碼。

一、獲取Bean

11張流程圖搞定 Spring Bean 生命週期

第一階段獲取Bean

這裡的流程圖的入口在 AbstractBeanFactory類的 doGetBean方法,這裡可以配合前面的 getBean方法分析文章進行閱讀。主要流程就是

1、先處理Bean 的名稱,因為如果以“&”開頭的Bean名稱表示獲取的是對應的FactoryBean對象;

2、從緩存中獲取單例Bean,有則進一步判斷這個Bean是不是在創建中,如果是的就等待創建完畢,否則直接返回這個Bean對象

3、如果不存在單例Bean緩存,則先進行循環依賴的解析

4、解析完畢之後先獲取父類BeanFactory,獲取到了則調用父類的getBean方法,不存在則先合併然後創建Bean

二、創建Bean

2.1 創建Bean之前

11張流程圖搞定 Spring Bean 生命週期

在真正創建Bean之前邏輯

這個流程圖對應的代碼在 AbstractAutowireCapableBeanFactory類的 createBean方法中。

1、這裡會先獲取 RootBeanDefinition對象中的Class對象並確保已經關聯了要創建的Bean的Class 。2、這裡會檢查3個條件

(1)Bean的屬性中的 beforeInstantiationResolved字段是否為true,默認是false。

(2)Bean是原生的Bean

(3)Bean的 hasInstantiationAwareBeanPostProcessors屬性為true,這個屬性在Spring準備刷新容器錢轉杯BeanPostProcessors的時候會設置,如果當前Bean實現了 InstantiationAwareBeanPostProcessor則這個就會是true。

當三個條件都存在的時候,就會調用實現的 InstantiationAwareBeanPostProcessor接口的 postProcessBeforeInstantiation方法,然後獲取返回的Bean,如果返回的Bean不是null還會調用實現的 BeanPostProcessor接口的 postProcessAfterInitialization方法,這裡用代碼說明


11張流程圖搞定 Spring Bean 生命週期


3、如果上面3個條件其中一個不滿足就不會調用實現的方法。默認這裡都不會調用的這些 BeanPostProcessors的實現方法。然後繼續執行後面的 doCreateBean方法。

2.1 真正的創建Bean,doCreateBean

11張流程圖搞定 Spring Bean 生命週期

doCreateBean方法邏輯

這個代碼的實現還是在 AbstractAutowireCapableBeanFactory方法中。流程是

1、先檢查 instanceWrapper變量是不是null,這裡一般是null,除非當前正在創建的Bean在 factoryBeanInstanceCache中存在這個是保存還沒創建完成的FactoryBean的集合。

2、調用createBeanInstance方法實例化Bean,這個方法在後面會講解

3、如果當前 RootBeanDefinition對象還沒有調用過實現了的 MergedBeanDefinitionPostProcessor接口的方法,則會進行調用 。

4、 當滿足以下三點(1)是單例Bean(2)嘗試解析bean之間的循環引用(3)bean目前正在創建中則會進一步檢查是否實現了 SmartInstantiationAwareBeanPostProcessor接口如果實現了則調用是實現的 getEarlyBeanReference方法

5、 調用 populateBean方法進行屬性填充,這裡後面會講解

6、 調用 initializeBean方法對Bean進行初始化,這裡後面會講解

2.1.1 實例化Bean,createBeanInstance

11張流程圖搞定 Spring Bean 生命週期

實例化Bean

這裡的邏輯稍微有一點複雜,這個流程圖已經是簡化過後的了。簡要根據代碼說明一下流程


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


1、先檢查Class是否已經關聯了,並且對應的修飾符是否是public的

2、如果用戶定義了Bean實例化的函數,則調用並返回

3、如果當前Bean實現了 FactoryBean接口則調用對應的 FactoryBean接口的 getObject方法

4、根據getBean時候是否傳入構造參數進行處理

4.1 如果沒有傳入構造參數,則檢查是否存在已經緩存的無參構造器,有則使用構造器直接創建,沒有就會調用 instantiateBean方法先獲取實例化的策略默認是 CglibSubclassingInstantiationStrategy,然後實例化Bean。最後返回

4.2 如果傳入了構造參數,則會先檢查是否實現了 SmartInstantiationAwareBeanPostProcessor接口,如果實現了會調用 determineCandidateConstructors獲取返回的候選構造器。

4.3

檢查4個條件是否滿足一個

(1)構造器不為null,

(2)從RootBeanDefinition中獲取到的關聯的注入方式是構造器注入(沒有構造參數就是setter注入,有則是構造器注入)

(3)含有構造參數

(4)getBean方法傳入構造參數不是空

滿足其中一個則會調用返回的候選構造器實例化Bean並返回,如果都不滿足,則會根據構造參數選擇合適的有參構造器然後實例化Bean並返回

5、如果上面都沒有合適的構造器,則直接使用無參構造器創建並返回Bean。

2.1.2 填充Bean,populateBean

11張流程圖搞定 Spring Bean 生命週期

填充Bean

這裡還是根據代碼來說一下流程


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


1、檢查當前Bean是否實現了 InstantiationAwareBeanPostProcessor的 postProcessAfterInstantiation方法則調用,並結束Bean的填充。2、將按照類型跟按照名稱注入的Bean分開,如果注入的Bean還沒有實例化的這裡會實例化,然後放到 PropertyValues對象中。3、如果實現了 InstantiationAwareBeanPostProcessor類的 postProcessProperties則調用這個方法並獲取返回值,如果返回值是null,則有可能是實現了過期的 postProcessPropertyValues方法,這裡需要進一步調用 postProcessPropertyValues方法4、進行參數填充

2.1.3 初始化Bean,initializeBean

11張流程圖搞定 Spring Bean 生命週期

初始化Bean

同時這裡根據代碼跟流程圖來說明

1、如果Bean實現了 BeanNameAware, BeanClassLoaderAware, BeanFactoryAware則調用對應實現的方法 。

2、Bean不為null並且bean不是合成的,如果實現了 BeanPostProcessor的 postProcessBeforeInitialization則會調用實現的 postProcessBeforeInitialization方法。在 ApplicationContextAwareProcessor類中實現了 postProcessBeforeInitialization方法。而這個類會在Spring刷新容器準備 beanFactory的時候會加進去,這裡就會被調用,而調用裡面會檢查Bean是不是 EnvironmentAware, EmbeddedValueResolverAware, ResourceLoaderAware, ApplicationEventPublisherAware, MessageSourceAware, ApplicationContextAware的實現類。這裡就會調用對應的實現方法。代碼如下


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


1、實例化Bean然後,檢查是否實現了 InitializingBean的 afterPropertiesSet方法,如果實現了就會調用

2、Bean不為null並且bean不是合成的,如果實現了 BeanPostProcessor的 postProcessBeforeInitialization則會調用實現的 postProcessAfterInitialization方法。

到此創建Bean 的流程就沒了,剩下的就是容器銷燬的時候的了

三、destory方法跟銷燬Bean

Bean在創建完畢之後會檢查用戶是否指定了 destroyMethodName以及是否實現了 DestructionAwareBeanPostProcessor接口的 requiresDestruction方法,如果指定了會記錄下來保存在 DisposableBeanAdapter對象中並保存在bean的 disposableBeans屬性中。代碼在 AbstractBeanFactory的 registerDisposableBeanIfNecessary中


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期

在銷燬Bean的時候最後都會調用 AbstractAutowireCapableBeanFactory的 destroyBean方法。


11張流程圖搞定 Spring Bean 生命週期

這裡是創建一個 DisposableBeanAdapter對象,這個對象實現了Runnable接口,在實現的 run方法中會調用實現的 DisposableBean接口的 destroy方法。並且在創建 DisposableBeanAdapter對象的時候會根據傳入的bean是否實現了 DisposableBean接口來設置 invokeDisposableBean變量,這個變量表實有沒有實現 DisposableBean接口


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


11張流程圖搞定 Spring Bean 生命週期


四、總結

最後來一個大的流程

11張流程圖搞定 Spring Bean 生命週期

實例化前的準備階段

11張流程圖搞定 Spring Bean 生命週期

實例化前

11張流程圖搞定 Spring Bean 生命週期

實例化後

11張流程圖搞定 Spring Bean 生命週期

初始化前

11張流程圖搞定 Spring Bean 生命週期

來源:www.jianshu.com/p/70b935f2b3fe


分享到:


相關文章: