Spring三級(jí)緩存思想解決循環(huán)依賴總結(jié)分析
三級(jí)緩存思想
Spring 解決循環(huán)依賴的核心就是提前暴露對(duì)象,而提前暴露的對(duì)象就是放置于第二級(jí)緩存中。下表是三級(jí)緩存的說(shuō)明:

所有被 Spring 管理的 Bean,最終都會(huì)存放在 singletonObjects 中,這里面存放的 Bean 是經(jīng)歷了所有生命周期的(除了銷毀的生命周期),完整的,可以給用戶使用的。
earlySingletonObjects 存放的是已經(jīng)被實(shí)例化,但是還沒(méi)有注入屬性和執(zhí)行 init 方法的 Bean。
singletonFactories 存放的是生產(chǎn) Bean 的工廠。
解決循環(huán)依賴
Spring 是如何通過(guò)上面介紹的三級(jí)緩存來(lái)解決循環(huán)依賴的呢?這里只用 A,B 形成的循環(huán)依賴來(lái)舉例:
- 實(shí)例化 A,此時(shí) A 還未完成屬性填充和初始化方法(@PostConstruct)的執(zhí)行,A 只是一個(gè)半成品。
- 為 A 創(chuàng)建一個(gè) Bean 工廠,并放入到 singletonFactories 中。
- 發(fā)現(xiàn) A 需要注入 B 對(duì)象,但是一級(jí)、二級(jí)、三級(jí)緩存均為發(fā)現(xiàn)對(duì)象 B。
- 實(shí)例化 B,此時(shí) B 還未完成屬性填充和初始化方法(@PostConstruct)的執(zhí)行,B 只是一個(gè)半成品。
- 為 B 創(chuàng)建一個(gè) Bean 工廠,并放入到 singletonFactories 中。
- 發(fā)現(xiàn) B 需要注入 A 對(duì)象,此時(shí)在一級(jí)、二級(jí)未發(fā)現(xiàn)對(duì)象 A,但是在三級(jí)緩存中發(fā)現(xiàn)了對(duì)象 A,從三級(jí)緩存中得到對(duì)象 A,并將對(duì)象 A 放入二級(jí)緩存中,同時(shí)刪除三級(jí)緩存中的對(duì)象 A。(注意,此時(shí)的 A 還是一個(gè)半成品,并沒(méi)有完成屬性填充和執(zhí)行初始化方法)
- 將對(duì)象 A 注入到對(duì)象 B 中。
- 對(duì)象 B 完成屬性填充,執(zhí)行初始化方法,并放入到一級(jí)緩存中,同時(shí)刪除二級(jí)緩存中的對(duì)象 B。(此時(shí)對(duì)象 B 已經(jīng)是一個(gè)成品)
- 對(duì)象 A 得到對(duì)象 B,將對(duì)象 B 注入到對(duì)象 A 中。(對(duì)象 A 得到的是一個(gè)完整的對(duì)象 B)
- 對(duì)象 A 完成屬性填充,執(zhí)行初始化方法,并放入到一級(jí)緩存中,同時(shí)刪除二級(jí)緩存中的對(duì)象 A。
我們從源碼中來(lái)分析整個(gè)過(guò)程:
創(chuàng)建 Bean 的方法在 AbstractAutowireCapableBeanFactory::doCreateBean()
protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, Object[] args) throws BeanCreationException {
BeanWrapper instanceWrapper = null;
if (instanceWrapper == null) {
// ① 實(shí)例化對(duì)象
instanceWrapper = this.createBeanInstance(beanName, mbd, args);
}
final Object bean = instanceWrapper != null ? instanceWrapper.getWrappedInstance() : null;
Class<?> beanType = instanceWrapper != null ? instanceWrapper.getWrappedClass() : null;
// ② 判斷是否允許提前暴露對(duì)象,如果允許,則直接添加一個(gè) ObjectFactory 到三級(jí)緩存
boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
// 添加三級(jí)緩存的方法詳情在下方
addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}
// ③ 填充屬性
this.populateBean(beanName, mbd, instanceWrapper);
// ④ 執(zhí)行初始化方法,并創(chuàng)建代理
exposedObject = initializeBean(beanName, exposedObject, mbd);
return exposedObject;
}添加三級(jí)緩存的方法如下:
protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
Assert.notNull(singletonFactory, "Singleton factory must not be null");
synchronized (this.singletonObjects) {
if (!this.singletonObjects.containsKey(beanName)) { // 判斷一級(jí)緩存中不存在此對(duì)象
this.singletonFactories.put(beanName, singletonFactory); // 添加至三級(jí)緩存
this.earlySingletonObjects.remove(beanName); // 確保二級(jí)緩存沒(méi)有此對(duì)象
this.registeredSingletons.add(beanName);
}
}
}
@FunctionalInterface
public interface ObjectFactory<T> {
T getObject() throws BeansException;
}通過(guò)這段代碼,我們可以知道 Spring 在實(shí)例化對(duì)象之后,就會(huì)為其創(chuàng)建一個(gè) Bean 工廠,并將此工廠加入到三級(jí)緩存中。
因此,Spring 一開(kāi)始提前暴露的并不是實(shí)例化的 Bean,而是將 Bean 包裝起來(lái)的 ObjectFactory。為什么要這么做呢?
這實(shí)際上涉及到 AOP,如果創(chuàng)建的 Bean 是有代理的,那么注入的就應(yīng)該是代理 Bean,而不是原始的 Bean。但是 Spring 一開(kāi)始并不知道 Bean 是否會(huì)有循環(huán)依賴,通常情況下(沒(méi)有循環(huán)依賴的情況下),Spring 都會(huì)在完成填充屬性,并且執(zhí)行完初始化方法之后再為其創(chuàng)建代理。但是,如果出現(xiàn)了循環(huán)依賴的話,Spring 就不得不為其提前創(chuàng)建代理對(duì)象,否則注入的就是一個(gè)原始對(duì)象,而不是代理對(duì)象。因此,這里就涉及到應(yīng)該在哪里提前創(chuàng)建代理對(duì)象?
Spring 的做法就是在 ObjectFactory 中去提前創(chuàng)建代理對(duì)象。它會(huì)執(zhí)行 getObject() 方法來(lái)獲取到 Bean。實(shí)際上,它真正執(zhí)行的方法如下:
protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) {
Object exposedObject = bean;
if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
for (BeanPostProcessor bp : getBeanPostProcessors()) {
if (bp instanceof SmartInstantiationAwareBeanPostProcessor) {
SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp;
// 如果需要代理,這里會(huì)返回代理對(duì)象;否則返回原始對(duì)象
exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName);
}
}
}
return exposedObject;
}因?yàn)樘崆斑M(jìn)行了代理,避免對(duì)后面重復(fù)創(chuàng)建代理對(duì)象,會(huì)在 earlyProxyReferences 中記錄已被代理的對(duì)象。
public abstract class AbstractAutoProxyCreator extends ProxyProcessorSupport
implements SmartInstantiationAwareBeanPostProcessor, BeanFactoryAware {
@Override
public Object getEarlyBeanReference(Object bean, String beanName) {
Object cacheKey = getCacheKey(bean.getClass(), beanName);
// 記錄已被代理的對(duì)象
this.earlyProxyReferences.put(cacheKey, bean);
return wrapIfNecessary(bean, beanName, cacheKey);
}
}通過(guò)上面的解析,我們可以知道 Spring 需要三級(jí)緩存的目的是為了在沒(méi)有循環(huán)依賴的情況下,延遲代理對(duì)象的創(chuàng)建,使 Bean 的創(chuàng)建符合 Spring 的設(shè)計(jì)原則。
如何獲取依賴
通過(guò)一個(gè) getSingleton() 方法去獲取所需要的 Bean 的。
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
// 一級(jí)緩存
Object singletonObject = this.singletonObjects.get(beanName);
if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
synchronized (this.singletonObjects) {
// 二級(jí)緩存
singletonObject = this.earlySingletonObjects.get(beanName);
if (singletonObject == null && allowEarlyReference) {
// 三級(jí)緩存
ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
if (singletonFactory != null) {
// Bean 工廠中獲取 Bean
singletonObject = singletonFactory.getObject();
// 放入到二級(jí)緩存中
this.earlySingletonObjects.put(beanName, singletonObject);
this.singletonFactories.remove(beanName);
}
}
}
}
return singletonObject;
}當(dāng) Spring 為某個(gè) Bean 填充屬性的時(shí)候,它首先會(huì)尋找需要注入對(duì)象的名稱,然后依次執(zhí)行 getSingleton() 方法得到所需注入的對(duì)象,而獲取對(duì)象的過(guò)程就是先從一級(jí)緩存中獲取,一級(jí)緩存中沒(méi)有就從二級(jí)緩存中獲取,二級(jí)緩存中沒(méi)有就從三級(jí)緩存中獲取,如果三級(jí)緩存中也沒(méi)有,那么就會(huì)去執(zhí)行 doCreateBean() 方法創(chuàng)建這個(gè) Bean。
二級(jí)緩存能夠解決循環(huán)依賴嗎
第三級(jí)緩存的目的是為了延遲代理對(duì)象的創(chuàng)建,因?yàn)槿绻麤](méi)有依賴循環(huán)的話,那么就不需要為其提前創(chuàng)建代理,可以將它延遲到初始化完成之后再創(chuàng)建。
既然目的只是延遲的話,那么我們是不是可以不延遲創(chuàng)建,而是在實(shí)例化完成之后,就為其創(chuàng)建代理對(duì)象,這樣我們就不需要第三級(jí)緩存了。因此,我們可以將addSingletonFactory() 方法進(jìn)行改造。
protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
Assert.notNull(singletonFactory, "Singleton factory must not be null");
synchronized (this.singletonObjects) {
if (!this.singletonObjects.containsKey(beanName)) { // 判斷一級(jí)緩存中不存在此對(duì)象
object o = singletonFactory.getObject(); // 直接從工廠中獲取 Bean
this.earlySingletonObjects.put(beanName, o); // 添加至二級(jí)緩存中
this.registeredSingletons.add(beanName);
}
}
}這樣的話,每次實(shí)例化完 Bean 之后就直接去創(chuàng)建代理對(duì)象,并添加到二級(jí)緩存中。測(cè)試結(jié)果是完全正常的,Spring 的初始化時(shí)間應(yīng)該也是不會(huì)有太大的影響,因?yàn)槿绻?Bean 本身不需要代理的話,是直接返回原始 Bean 的,并不需要走復(fù)雜的創(chuàng)建代理 Bean 的流程。
測(cè)試證明,二級(jí)緩存也是可以解決循環(huán)依賴的。為什么 Spring 不選擇二級(jí)緩存,而要額外多添加一層緩存呢?
如果 Spring 選擇二級(jí)緩存來(lái)解決循環(huán)依賴的話,那么就意味著所有 Bean 都需要在實(shí)例化完成之后就立馬為其創(chuàng)建代理,而 Spring 的設(shè)計(jì)原則是在 Bean 初始化完成之后才為其創(chuàng)建代理。所以,Spring 選擇了三級(jí)緩存。但是因?yàn)檠h(huán)依賴的出現(xiàn),導(dǎo)致了 Spring 不得不提前去創(chuàng)建代理,因?yàn)槿绻惶崆皠?chuàng)建代理對(duì)象,那么注入的就是原始對(duì)象,這樣就會(huì)產(chǎn)生錯(cuò)誤。
解決循環(huán)依賴總結(jié)
Spring設(shè)計(jì)了三級(jí)緩存來(lái)解決循環(huán)依賴問(wèn)題。
- 第一級(jí)緩存里面存儲(chǔ)完整的bean實(shí)例,這些實(shí)例是可以直接被使用的;
- 第二級(jí)緩存里面存儲(chǔ)的實(shí)例化以后但是還沒(méi)有設(shè)置屬性值的bean實(shí)例,也就是bean里面的依賴注入還沒(méi)有做;
- 第三級(jí)緩存用來(lái)存放bean工廠,它主要用來(lái)生成原始bean對(duì)象,并且放到第二個(gè)緩存里面。
三級(jí)緩存的核心思想就是把bean的實(shí)例化和bean里面的依賴注入進(jìn)行分離,采用一級(jí)緩存存儲(chǔ)完整的bean實(shí)例,采用二級(jí)緩存來(lái)儲(chǔ)存不完整的bean實(shí)例。通過(guò)不完整的bean實(shí)例作為突破口,解決循環(huán)依賴問(wèn)題。至于第三級(jí)緩存,主要是解決代理對(duì)象的循環(huán)依賴問(wèn)題。
spring無(wú)法解決的循環(huán)依賴場(chǎng)景
- 多實(shí)例的Setter注入導(dǎo)致的循環(huán)依賴,需要把bean改成單例。
- 構(gòu)造器注入導(dǎo)致的循環(huán)依賴,可以通過(guò)@Lazy注解。
- DependsOn導(dǎo)致的循環(huán)依賴,找到注解循環(huán)依賴的地方,使它不循環(huán)依賴。
- 單例代理對(duì)象Setter注入導(dǎo)致的循環(huán)依賴,可以使用 @Lazy注解;或者使用 @DependsOn注解指定加載先后關(guān)系。
多例、構(gòu)造器注入為什么不能解決循環(huán)依賴?
因?yàn)檠h(huán)依賴的原理是實(shí)例化后提前暴露的引用,這兩種情況還沒(méi)實(shí)例化
以上就是Spring三級(jí)緩存思想解決循環(huán)依賴總結(jié)分析的詳細(xì)內(nèi)容,更多關(guān)于Spring解決循環(huán)依賴的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Spring中的三級(jí)緩存與循環(huán)依賴詳解
- Spring的循環(huán)依賴、三級(jí)緩存解決方案源碼詳細(xì)解析
- Spring通過(guò)三級(jí)緩存解決循環(huán)依賴問(wèn)題的過(guò)程詳解
- Java中通過(guò)三級(jí)緩存解決Spring循環(huán)依賴詳解
- 一文教你如何通過(guò)三級(jí)緩存解決Spring循環(huán)依賴
- Spring三級(jí)緩存解決循環(huán)依賴的過(guò)程分析
- 教你Spring如何使用三級(jí)緩存解決循環(huán)依賴
- 一篇文章帶你理解Java Spring三級(jí)緩存和循環(huán)依賴
- Spring為什么要用三級(jí)緩存解決循環(huán)依賴呢
相關(guān)文章
使用SpringBoot開(kāi)發(fā)Restful服務(wù)實(shí)現(xiàn)增刪改查功能
Spring Boot是由Pivotal團(tuán)隊(duì)提供的全新框架,其設(shè)計(jì)目的是用來(lái)簡(jiǎn)化新Spring應(yīng)用的初始搭建以及開(kāi)發(fā)過(guò)程。這篇文章主要介紹了基于SpringBoot開(kāi)發(fā)一個(gè)Restful服務(wù),實(shí)現(xiàn)增刪改查功能,需要的朋友可以參考下2018-01-01
java發(fā)送get請(qǐng)求和post請(qǐng)求示例
這篇文章主要介紹了java發(fā)送get請(qǐng)求和post請(qǐng)求示例,需要的朋友可以參考下2014-03-03
java四種訪問(wèn)權(quán)限實(shí)例分析
這篇文章主要介紹了java四種訪問(wèn)權(quán)限實(shí)例分析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
SpringBoot YAML語(yǔ)法基礎(chǔ)詳細(xì)整理
YAML 是 “YAML Ain’t Markup Language”(YAML 不是一種標(biāo)記語(yǔ)言)的遞歸縮寫。在開(kāi)發(fā)的這種語(yǔ)言時(shí),YAML 的意思其實(shí)是:“Yet Another Markup Language”(仍是一種標(biāo)記語(yǔ)言),本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-10-10
分布式服務(wù)Dubbo+Zookeeper安全認(rèn)證實(shí)例
下面小編就為大家分享一篇分布式服務(wù)Dubbo+Zookeeper安全認(rèn)證實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-12-12

