Spring压轴题:当循环依赖遇上Spring AOP
Spring壓軸題:當(dāng)循環(huán)依賴遇上Spring AOP
前言
問:Spring如何解決循環(huán)依賴?
答:Spring通過提前曝光機(jī)制,利用三級緩存解決循環(huán)依賴(這原理還是挺簡單的,參考:三級緩存、圖解循環(huán)依賴原理)
再問:Spring通過提前曝光,直接曝光到二級緩存已經(jīng)可以解決循環(huán)依賴問題了 ,為什么一定要三級緩存?
再細(xì)問:如果循環(huán)依賴的時(shí)候 ,所有類又都需要Spring AOP自動代理,那Spring如何提前曝光?曝光的是原始bean還是代理后的bean?
這些問題算是Spring源碼的壓軸題了,如果這些問題都弄明白,恭喜你順利結(jié)業(yè)Spring源碼了。就單單對Spring這一塊的理解,不夸張的說可以達(dá)到阿里水準(zhǔn)了
源碼分析
進(jìn)入正題,在Spring創(chuàng)建Bean的核心代碼doGetBean中,在實(shí)例化bean之前,會先嘗試從三級緩存獲取bean,這也是Spring解決循環(huán)依賴的開始
(一) 緩存中獲取bean
// AbstractBeanFactory.javaprotected T doGetBean(final String name, @Nullable final ClassrequiredType, @Nullable final Object[] args, boolean typeCheckOnly) throws BeansException { final String beanName = transformedBeanName(name); Object bean; // 2. 嘗試從緩存中獲取bean Object sharedInstance = getSingleton(beanName); ...} protected Object getSingleton(String beanName, boolean allowEarlyReference) { // 從一級緩存獲取 ,key=beanName value=bean Object singletonObject = this.singletonObjects.get(beanName); if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) { synchronized (this.singletonObjects) { // 從二級緩存獲取,key=beanName value=bean singletonObject = this.earlySingletonObjects.get(beanName); // 是否允許循環(huán)引用 if (singletonObject == null && allowEarlyReference) { /** * 三級緩存獲取
,key=beanName value=objectFactory,objectFactory中存儲getObject()方法用于獲取提前曝光的實(shí)例 * * 而為什么不直接將實(shí)例緩存到二級緩存,而要多此一舉將實(shí)例先封裝到objectFactory中
? * 主要關(guān)鍵點(diǎn)在getObject()方法并非直接返回實(shí)例,而是對實(shí)例又使用 * SmartInstantiationAwareBeanPostProcessor的getEarlyBeanReference方法對bean進(jìn)行處理 * * 也就是說,當(dāng)spring中存在該后置處理器 ,所有的單例bean在實(shí)例化后都會被進(jìn)行提前曝光到三級緩存中, * 但是并不是所有的bean都存在循環(huán)依賴
,也就是三級緩存到二級緩存的步驟不一定都會被執(zhí)行,有可能曝光后直接創(chuàng)建完成 ,沒被提前引用過