欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深度解析SpringBoot中@Async引起的循環(huán)依賴

 更新時(shí)間:2022年02月09日 16:40:46   作者:程銘程銘你快成名  
本文主要介紹了深度解析SpringBoot中@Async引起的循環(huán)依賴,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

啊,昨晚發(fā)版又出現(xiàn)了讓有頭大的循環(huán)依賴問題,按理說Spring會為我們解決循環(huán)依賴,但是為什么還會出現(xiàn)這個問題呢?為什么在本地、UAT以及PRE環(huán)境都沒有出現(xiàn)這個問題,但是到了PROD環(huán)境就出現(xiàn)了這個問題呢?本文將從事故時(shí)間線、及時(shí)止損、復(fù)盤分析等幾個方面為大家?guī)碓敿?xì)的分析,干貨滿滿!

事故時(shí)間線

本著"先止損、后復(fù)盤分析"的原則,我們來看一下這次發(fā)版事故的時(shí)間線。

2021年11月16日晚23點(diǎn)00分00秒開始發(fā)版,此時(shí)集團(tuán)的devops有點(diǎn)慢

2021年11月16日晚23點(diǎn)03分01秒,收到發(fā)版失敗的消息,登錄服務(wù)器發(fā)現(xiàn)發(fā)生了循環(huán)依賴,具體錯誤如下圖,從日志中可以看到是dataCollectionSendMessageService這個bean出現(xiàn)了循環(huán)依賴

在這里插入圖片描述

問題發(fā)現(xiàn)了就需要先解決,然后再去分析為什么??吹竭@個報(bào)錯日志我心里也大概知道是為什么了,所以很快就解決了,解決方案如下:給DataCollectionSendMessageService加上@Lazy注解

在這里插入圖片描述

2021年11月16日晚23點(diǎn)07分16秒,使用重新集成的代碼開始發(fā)版,大概10分鐘后線上節(jié)點(diǎn)全部發(fā)版完成。從時(shí)間線來看從發(fā)現(xiàn)問題到解決問題,前后一共用了接近15分鐘(這期間代碼集成和發(fā)布用了過多的時(shí)間),也算是做到了及時(shí)止損,沒有讓問題繼續(xù)擴(kuò)大。

猜想

我大膽的猜想是因?yàn)榇蛄?code>@Aysnc注解的bean生成了對象的代理,導(dǎo)致Spring bean最終加載的不是一個原始對象導(dǎo)致了此次問題的發(fā)生,那么對不對呢,接下來我們通過源碼詳細(xì)分析一下。

什么是循環(huán)依賴

所謂循環(huán)依賴就是Spring IOC容器在加載bean時(shí)會按照順序加載,先去實(shí)例化 beanA。然后發(fā)現(xiàn) beanA 依賴于 beanB,接在又去實(shí)例化 beanB。實(shí)例化 beanB 時(shí),發(fā)現(xiàn) beanB 又依賴于 beanA。如果容器不處理循環(huán)依賴的話,容器會無限執(zhí)行上面的流程,直到內(nèi)存溢出,程序崩潰,所以這個時(shí)候就會拋出BeanCurrentlyInCreationException異常,也就是我們常說的循環(huán)依賴,下面是兩種常見循環(huán)依賴的場景。

幾個Bean之間的循環(huán)依賴

@Component
public class A {

    @Autowired
    private B b;
}

@Component
public class B {

    @Autowired
    private C c;
}

@Component
public class C {

    @Autowired
    private A a;
}

效果圖如下:

在這里插入圖片描述

自己依賴自己

@Component
public class A {

    @Autowired
    private A a;
}

效果圖如下:

在這里插入圖片描述

Spring是如何解決循環(huán)依賴的

在這里插入圖片描述

首先Spring維護(hù)了三個Map,也就是我們通常說的三級緩存

  • singletonObjects:俗稱單例池,緩存創(chuàng)建完成的單例Bean
  • singletonFactories:映射創(chuàng)建Bean的原始工廠
  • earlySingletonObjects:映射Bean的早期引用,也就是說這個Map里的Bean不是完整的,只是完成了實(shí)例化,但還沒有初始化

Spring通過三級緩存解決了循環(huán)依賴,其中一級緩存為單例池(singletonObjects),二級緩存為早期曝光對象earlySingletonObjects,三級緩存為早期曝光對象工廠(singletonFactories)。

當(dāng)A、B兩個類發(fā)生循環(huán)引用時(shí),在A完成實(shí)例化后,就使用實(shí)例化后的對象去創(chuàng)建一個對象工廠,并添加到三級緩存中,如果A被AOP代理,那么通過這個工廠獲取到的就是A代理后的對象,如果A沒有被AOP代理,那么這個工廠獲取到的就是A實(shí)例化的對象。

當(dāng)A進(jìn)行屬性注入時(shí),會去創(chuàng)建B,同時(shí)B又依賴了A,所以創(chuàng)建B的同時(shí)又會去調(diào)用getBean(a)來獲取需要的依賴,此時(shí)的getBean(a)會從緩存中獲取,第一步,先獲取到三級緩存中的工廠;第二步,調(diào)用對象工工廠的getObject方法來獲取到對應(yīng)的對象,得到這個對象后將其注入到B中。

緊接著B會走完它的生命周期流程,包括初始化、后置處理器等。當(dāng)B創(chuàng)建完后,會將B再注入到A中,此時(shí)A再完成它的整個生命周期。至此,循環(huán)依賴結(jié)束!

簡單一句話說:先去緩存里找Bean,沒有則實(shí)例化當(dāng)前的Bean放到Map,如果有需要依賴當(dāng)前Bean的,就能從Map取到。

什么是@Async

@Async注解是Spring為我們提供的異步調(diào)用的注解,@Async可以作用到類或者方法上,標(biāo)記了@Async注解的方法將會在獨(dú)立的線程中被執(zhí)行,調(diào)用者無需等待它的完成,即可繼續(xù)其他的操作。從源碼中可以看到標(biāo)記了@Async注解的方法會被提交到org.springframework.core.task.TaskExecutor中異步執(zhí)行。

在這里插入圖片描述


或者我們可以通過value來指定使用哪個自定義線程池,比如這樣子:

@Async("asyncTaskExecutor")

被@Async標(biāo)記的bean注入時(shí)機(jī)

我們從源碼的角度來看一下被@Async標(biāo)記的bean是如何注入到Spring容器里的。在我們開啟@EnableAsync注解之后代表可以向Spring容器中注入AsyncAnnotationBeanPostProcessor,它是一個后置處理器,我們看一下他的類圖。

在這里插入圖片描述

真正創(chuàng)建代理對象的代碼在AbstractAdvisingBeanPostProcessor中的postProcessAfterInitialization方法中,以下代碼有所刪減,只保留核心邏輯代碼

	// 這個map用來緩存所有被postProcessAfterInitialization這個方法處理的bean
	private final Map<Class<?>, Boolean> eligibleBeans = new ConcurrentHashMap<>(256);

	// 這個方法主要是為打了@Async注解的bean生成代理對象
	@Override
	public Object postProcessAfterInitialization(Object bean, String beanName) {
		// 這里是重點(diǎn),這里返回true
		if (isEligible(bean, beanName)) {
			// 工廠模式生成一個proxyFactory
			ProxyFactory proxyFactory = prepareProxyFactory(bean, beanName);
			if (!proxyFactory.isProxyTargetClass()) {
				evaluateProxyInterfaces(bean.getClass(), proxyFactory);
			}
			// 切入切面并創(chuàng)建一個代理對象
			proxyFactory.addAdvisor(this.advisor);
			customizeProxyFactory(proxyFactory);
			return proxyFactory.getProxy(getProxyClassLoader());
		}
		// No proxy needed.
		return bean;
	}
	protected boolean isEligible(Class<?> targetClass) {
		// 首次從eligibleBeans這個map中一定是拿不到的
		Boolean eligible = this.eligibleBeans.get(targetClass);
		if (eligible != null) {
			return eligible;
		}
		// 如果沒有advisor,也就是切面,直接返回false
		if (this.advisor == null) {
			return false;
		}
		// 這里判斷AsyncAnnotationAdvisor能否切入,因?yàn)槲覀兊腷ean是打了@Aysnc注解,這里是一定能切入的,最終會返回true
		eligible = AopUtils.canApply(this.advisor, targetClass);
		this.eligibleBeans.put(targetClass, eligible);
		return eligible;
	}

至此打了@Aysnc注解的bean就創(chuàng)建完成了,結(jié)果是生成了一個代理對象

循環(huán)依賴到底是怎么生成的

經(jīng)過上面的源碼分析,我們可以知道有@Aysnc注解的bean最后生成了一個代理對象,我們結(jié)合Spring bean創(chuàng)建的流程來分析這次問題。

  • beanA開始初始化,beanA實(shí)例化完成后給beanA的依賴屬性beanB進(jìn)行賦值
  • beanB開始初始化,beanB實(shí)例化完成后給beanB的依賴屬性beanA進(jìn)行賦值
  • 因?yàn)?code>beanA是支持循環(huán)依賴的,所以可以在earlySingletonObjects中可以拿到beanA的早期引用的,但是因?yàn)?code>beanB打了@Aysnc注解并不能在earlySingletonObjects中可以拿到早期引用
  • 接下來執(zhí)行執(zhí)行initializeBean(Object existingBean, String beanName)方法,這里beanA可以正常實(shí)例化完成,但是因?yàn)?code>beanB打了@Aysnc注解,所以向Spring IOC容器中增加了一個代理對象,也就是說beanAbeanB并不是一個原始對象,而是一個代理對象
  • 接下來進(jìn)行執(zhí)行doCreateBean方法時(shí)對進(jìn)行檢測,以下代碼有所刪減,只保留核心邏輯代碼
protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)
			throws BeanCreationException {

		if (earlySingletonExposure) {
			Object earlySingletonReference = getSingleton(beanName, false);
			if (earlySingletonReference != null) {
				if (exposedObject == bean) {
					exposedObject = earlySingletonReference;
				}
				else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {
					String[] dependentBeans = getDependentBeans(beanName);
					Set<String> actualDependentBeans = new LinkedHashSet<>(dependentBeans.length);
					// 重點(diǎn)在這里,這里會遍歷所有依賴的bean,如果beanA依賴beanB和緩存中的beanB不相等
					// 也就是說beanA本來依賴的是一個原始對象beanB,但是這個時(shí)候發(fā)現(xiàn)beanB是一個代理對象,就會增加到actualDependentBeans
					for (String dependentBean : dependentBeans) {
						if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {
							actualDependentBeans.add(dependentBean);
						}
					}
					// 發(fā)現(xiàn)actualDependentBeans不為空,就發(fā)生了我們最開始截圖的錯誤
					if (!actualDependentBeans.isEmpty()) {
						throw new BeanCurrentlyInCreationException(beanName,
								"Bean with name '" + beanName + "' has been injected into other beans [" +
								StringUtils.collectionToCommaDelimitedString(actualDependentBeans) +
								"] in its raw version as part of a circular reference, but has eventually been " +
								"wrapped. This means that said other beans do not use the final version of the " +
								"bean. This is often the result of over-eager type matching - consider using " +
								"'getBeanNamesOfType' with the 'allowEagerInit' flag turned off, for example.");
					}
				}
			}
		}

		// Register bean as disposable.
		try {
			registerDisposableBeanIfNecessary(beanName, bean, mbd);
		}
		catch (BeanDefinitionValidationException ex) {
			throw new BeanCreationException(
					mbd.getResourceDescription(), beanName, "Invalid destruction signature", ex);
		}

		return exposedObject;
	}

解決循環(huán)依賴的正確姿勢

  • @Lazy注解
  • 代碼優(yōu)化,不要讓@Async的Bean參與循環(huán)依賴

至此我們就知道為什么發(fā)生了此次問題。

到此這篇關(guān)于深度解析SpringBoot中@Async引起的循環(huán)依賴的文章就介紹到這了,更多相關(guān)SpringBoot中@Async循環(huán)依賴內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java中的MapStruct知識點(diǎn)總結(jié)

    Java中的MapStruct知識點(diǎn)總結(jié)

    這篇文章主要介紹了Java中的MapStruct知識點(diǎn)總結(jié),MapStruct是一個Java注解處理器,用于生成類型安全的映射代碼,它可以自動處理源對象和目標(biāo)對象之間的映射,減少了手動編寫重復(fù)的映射代碼的工作量,需要的朋友可以參考下
    2023-10-10
  • SpringBoot整合高德地圖天氣查詢的全過程

    SpringBoot整合高德地圖天氣查詢的全過程

    這篇文章主要給大家介紹了關(guān)于SpringBoot整合高德地圖天氣查詢的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用springboot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2021-12-12
  • JavaSwing后臺播放音樂mp3

    JavaSwing后臺播放音樂mp3

    這篇文章主要為大家詳細(xì)介紹了JavaSwing后臺播放音樂mp3,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-06-06
  • 圖解分析Javaweb進(jìn)程與線程

    圖解分析Javaweb進(jìn)程與線程

    這篇文章主要介紹了Javaweb進(jìn)程與線程的知識,本篇文章通過簡要的案例,講解了它的基礎(chǔ)原理與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2022-03-03
  • Spring Boot + Jpa(Hibernate) 架構(gòu)基本配置詳解

    Spring Boot + Jpa(Hibernate) 架構(gòu)基本配置詳解

    本篇文章主要介紹了Spring Boot + Jpa(Hibernate) 架構(gòu)基本配置詳解,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • 基于Java寫minio客戶端實(shí)現(xiàn)上傳下載文件

    基于Java寫minio客戶端實(shí)現(xiàn)上傳下載文件

    這篇文章主要介紹了基于Java寫minio客戶端實(shí)現(xiàn)上傳下載文件,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-05-05
  • java關(guān)鍵字static學(xué)習(xí)心得

    java關(guān)鍵字static學(xué)習(xí)心得

    本篇文章給大家分享一篇關(guān)于java關(guān)鍵字static的學(xué)習(xí)心得,有這方面需要的朋友學(xué)習(xí)下吧。
    2018-01-01
  • JVM創(chuàng)建對象及訪問定位過程詳解

    JVM創(chuàng)建對象及訪問定位過程詳解

    這篇文章主要介紹了JVM創(chuàng)建對象及訪問定位過程詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-12-12
  • 一個簡單的Java音樂播放器

    一個簡單的Java音樂播放器

    這篇文章主要為大家分享一個簡單的Java音樂播放器,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • Java擴(kuò)展Nginx之共享內(nèi)存

    Java擴(kuò)展Nginx之共享內(nèi)存

    這篇文章主要介紹了Java擴(kuò)展Nginx之共享內(nèi)存的相關(guān)資料,需要的朋友可以參考下
    2023-07-07

最新評論