spring源碼學(xué)習(xí)之bean的初始化以及循環(huán)引用
實(shí)例化方法,把bean實(shí)例化,并且包裝成BeanWrapper
1、點(diǎn)進(jìn)這個方法里面。
這個方法是反射調(diào)用類中的 factoryMethod 方法。 這要知道@Bean 方法的原理, 實(shí)際上spring 會掃描有@bean 注解的方法, 然后把方法名稱設(shè)置到 BeanDefinition 的 factoryMethod屬性中, 接下來就會調(diào)到上面截圖中的方法實(shí)現(xiàn)@Bean 方法的調(diào)用。
2、 有參構(gòu)造函數(shù)的時(shí)候
determineConstructorsFromBeanPostProcessors這個方法是 BeanPostProcessor 接口類的首次應(yīng)用, 最終會掉到
AutowiredAnnotationBeanPostProcessor 類的方法, 在方法中會掃描有注解的構(gòu)造函數(shù)然后完成裝配過程。 然后把有有@Autowired 注解的構(gòu)造函數(shù)返回。
3、 無參構(gòu)造函數(shù)的實(shí)例化
這就是簡單的反射實(shí)例化。 大部分類的實(shí)例化都會走這個邏輯
4、 類中注解的收集
實(shí)例化完成后接下來就需要對類中的屬性進(jìn)行依賴注入操作, 但是類里面屬性和方法的依賴注入往往用@Autowired 或者@Resource 注解, 那么這些注解的依賴注入是如何完成的呢?
注解的收集:
也是通過 BeanPostProcessor 接口類型實(shí)例來挨個處理的。
A、 首先是CommonAnnotationBeanPostProcessor 類, 這個類完成了@Resource 注解的屬性或者方法的收集
這個類還對@PostConstruct 和@PreDestory 支持
收集過程
1、 看緩存里面有沒有 InjectionMetadata 對象
2、 從類中獲取所有 Field 對象, 循環(huán) field 對象, 判斷 field 有沒有@Resource 注解,如果有注解封裝成 ResourceElement 對象
3、 從類中獲取所有 Method 對象, 循環(huán) Method 對象, 判斷 Method 有沒有@Resource
注解, 如果有注解封裝成 ResourceElement 對象
4、 最終把兩個 field 和 Method 封裝的對象集合封裝到 InjectionMetadata 對象中
B、 然后是AutowiredAnnotationBeanPostProcessor 類, 對@Autowired 注解的屬性和方法的收集。 收集過程基本上跟@Resource 注解的收集差不多, 這里就不贅述了
5、 IOC\DI 依
對應(yīng)的方法:
這里又是一個 BeanPostProcessor 類型接口的運(yùn)用, 前面我們講到了@Resource@Autowired 注解的收集, 那么這個方法就是根據(jù)收集到的注解進(jìn)行反射調(diào)用。
循環(huán)收集到的 metaData 中的 list 對象, 然后挨個調(diào)用里面的 InjectedElement 的inject 方法完成依賴注入。
其中 value 值的獲取, 如果依賴的屬性是一個引用類型必定會觸發(fā)該屬性的BeanFactory.getBean 操作, 從而從 spring 容器中獲取到對應(yīng)的實(shí)例。 方法的依賴注入類似這里就不再贅述。
6、 bean 實(shí)例化后的操作
代碼走到這里:
A、 首先是對某些 Aware 接口的調(diào)用
B、 然后@PostConstruct 注解方法的調(diào)用
這里又是一個 BeanPostProcessor 接口的運(yùn)用,前面講過, 有@PostConstruct 注解的方法會收集到一個 met就是通過 BeanPostProcessor 接口調(diào)到CommonAnnotationBeanPostProcessor 類, 然后在類中拿根據(jù)對象里面的容器來反射調(diào)用有注解的方法。 代碼如下:
有@PostConstruct 注解的容器會收集到 initMethods 容器中, 接下來就是方法的
反射調(diào)用。
C、 InitializingBean 接口和 init-method 屬性調(diào)用
Init-method 屬性調(diào)用是在 afterPropertiesSet 之后
afterPropertiesSet和Init-method和有@PostConstruct注解的方法其實(shí)核心功能都是一樣的, 只是調(diào)用時(shí)序不一樣而已, 都是在該類實(shí)例化和 IOC 做完后調(diào)用的, 我們可以在這些方法中做一些在 spring 或者 servlet 容器啟動的時(shí)候的初始化工作。 比如緩存預(yù)熱, 比如緩存數(shù)據(jù)加載到內(nèi)存, 比如配置解析, 等等初始化工作。
在這個方法里面還有一個重要的邏輯
也是一個 BeanPostProcessor 接口的運(yùn)用, 在這里會返回 bean 的代理實(shí)例, 這個就是 AOP 的入口。
D、 FactoryBean 接口
帶入如下:
在實(shí)例化和 IOC/DI 做完后, 就會調(diào)用 FactoryBean 類型的接口, 如果要獲取到FactoryBean 類本身, 就必須加上”&”符號, 比如beanFactory.getBean(“&beanName” )。
BeanFactory.getBean(“beanName”)只能獲取到getObject()方法返回的實(shí)例。
getObject方法返回的實(shí)例會有單獨(dú)的緩存存儲, 跟其他實(shí)例不是同一個緩存, 對應(yīng)的緩存是:factoryBeanObjectCache
E、 循環(huán)依賴
循環(huán)依賴請參照流程圖理解https://www.processon.com/view/link/5df9ce52e4b0c4255ea1a84f
循環(huán)依賴只會出現(xiàn)在單例實(shí)例無參構(gòu)造函數(shù)實(shí)例化情況下
有參構(gòu)造函數(shù)的加@Autowired 的方式循環(huán)依賴是直接報(bào)錯的, 多例的循環(huán)依賴也是直接報(bào)錯的
循環(huán)依賴步驟:
1、 A 類無參構(gòu)造函數(shù)實(shí)例化后, 設(shè)置三級緩存
2、 A 類 populateBean 進(jìn)行依賴注入, 這里觸發(fā)了 B 類屬性的 getBean 操作
3、 B 類無參構(gòu)造函數(shù)實(shí)例化后, 設(shè)置三級緩存
4、 B 類 populateBean 進(jìn)行依賴注入, 這里觸發(fā)了 A 類屬性的 getBean 操作
5、 A 類之前正在實(shí)例化, singletonsCurrentlyInCreation 集合中有已經(jīng)有這個 A 類了, 三級緩存里面也有了, 所以這時(shí)候是從三級緩存中拿到的提前暴露的A 實(shí)例, 該實(shí)例還沒有進(jìn)行 B 類屬性的依賴注入的, B 類屬性為空。
6、 B 類拿到了 A 的提前暴露實(shí)例注入到 A 類屬性中了
7、 B 類實(shí)例化已經(jīng)完成, B 類的實(shí)例化是由 A 類實(shí)例化中 B 屬性的依賴注入觸發(fā)的 getBean 操作進(jìn)行的, 現(xiàn)在 B 已經(jīng)實(shí)例化, 所以 A 類中 B 屬性就可以完成依賴注入了, 這時(shí)候 A 類 B 屬性已經(jīng)有值了
8、 B 類 A 屬性指向的就是 A 類實(shí)例堆空間, 所以這時(shí)候 B 類 A 屬性也會有值了。
總結(jié)
到此這篇關(guān)于spring源碼學(xué)習(xí)之bean的初始化以及循環(huán)引用的文章就介紹到這了,更多相關(guān)spring源碼bean的初始化及循環(huán)引用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot集成mybatis-maven插件自動生成pojo的詳細(xì)教程
這篇文章主要介紹了springboot集成mybatis-maven插件自動生成pojo的詳細(xì)教程,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01關(guān)于Mybatis中foreach遍歷Map的實(shí)現(xiàn)示例
這篇文章主要介紹了關(guān)于Mybatis中foreach遍歷Map的實(shí)現(xiàn)示例,MyBatis?是一款優(yōu)秀的半自動的ORM持久層框架,它支持自定義?SQL、存儲過程以及高級映射,需要的朋友可以參考下2023-05-05MyBatisPlus分頁的同時(shí)指定排序規(guī)則說明
這篇文章主要介紹了MyBatisPlus分頁的同時(shí)指定排序規(guī)則說明,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12springboot項(xiàng)目實(shí)現(xiàn)斷點(diǎn)續(xù)傳功能
這篇文章主要介紹了springboot項(xiàng)目實(shí)現(xiàn)斷點(diǎn)續(xù)傳,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-08-08SpringBoot參數(shù)驗(yàn)證的幾種方式小結(jié)
在日常的接口開發(fā)中,為了防止非法參數(shù)對業(yè)務(wù)造成影響,經(jīng)常需要對接口的參數(shù)進(jìn)行校驗(yàn),例如登錄的時(shí)候需要校驗(yàn)用戶名和密碼是否為空,所以本文介紹了SpringBoot參數(shù)驗(yàn)證的幾種方式,需要的朋友可以參考下2024-07-07深入了解Java中Synchronized關(guān)鍵字的實(shí)現(xiàn)原理
synchronized是JVM的內(nèi)置鎖,基于Monitor機(jī)制實(shí)現(xiàn),每一個對象都有一個與之關(guān)聯(lián)的監(jiān)視器?(Monitor),這個監(jiān)視器充當(dāng)了一種互斥鎖的角色,本文就詳細(xì)聊一聊Synchronized關(guān)鍵字的實(shí)現(xiàn)原理,需要的朋友可以參考下2023-06-06關(guān)于spring data jpa一級緩存的問題
這篇文章主要介紹了關(guān)于spring data jpa一級緩存的問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-11-11