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

Java @Async注解導(dǎo)致spring啟動(dòng)失敗解決方案詳解

 更新時(shí)間:2021年09月02日 09:48:18   作者:liangsheng_g  
這篇文章主要介紹了Java @Async注解導(dǎo)致spring啟動(dòng)失敗解決方案詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下

前言

這篇文章里,最后總結(jié)處,我說了會(huì)講講循環(huán)依賴中,其中一個(gè)類添加@Async有可能會(huì)導(dǎo)致注入失敗而拋異常的情況,今天就分析一下。

一、異常表現(xiàn),拋出內(nèi)容

1.1循環(huán)依賴的兩個(gè)class

1.CycleService1

@Service
public class CycleService1 {

	@Autowired
	private CycleService2 cycleService2;

	@WangAnno
	@Async
	public void doThings() {
		System.out.println("it's a async move");
	}

}

2.CycleService2

@Service
public class CycleService2 {

	private CycleService1 cycleService1;

	public void init() {

	}

	@WangAnno
	public void alsoDo() {
		System.out.println("create cycleService2");
	}

}

1.2 啟動(dòng)報(bào)錯(cuò)

Bean with name ‘cycleService1' has been injected into other beans [cycleService2] 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.

警告: Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'cycleService1': Bean with name 'cycleService1' has been injected into other beans [cycleService2] 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.

Exception in thread "main" org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'cycleService1': Bean with name 'cycleService1' has been injected into other beans [cycleService2] 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.

at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:654)

at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:523)

at org.springframework.beans.factory.support.AbstractBeanFactory.lambda$doGetBean$0(AbstractBeanFactory.java:323)

at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:226)

at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:320)

at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:199)

at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:851)

at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:884)

at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:552)

at com.wang.Test.main(Test.java:109)

二、原因分析

2.1 主要原因

想想我在這篇博客里的圖解步驟:

  1. 當(dāng)Spring在進(jìn)行bean的實(shí)例化的時(shí)候,由于CycleService1和CycleService2是循環(huán)依賴的,
  2. 同時(shí),由于CycleService1創(chuàng)建早于CycleService2。
  3. 所以,在CycleService1對(duì)CycleService2的initializeBean方法執(zhí)行之后得到了exposedObject,要從二級(jí)緩存里獲取CycleService1的earlySingletonReference不為null,就需要比較exposedObject和raw CycleService是否還是同一個(gè)對(duì)象,如果不再是同一個(gè)對(duì)象,那么就會(huì)報(bào)錯(cuò)。
  4. 為什么有這個(gè)邏輯呢?
  5. 其實(shí)是因?yàn)槿绻軓亩?jí)緩存里拿出的earlySingletonReference不為null,說明了在該對(duì)象再創(chuàng)建過程中被其他對(duì)象循環(huán)依賴了,且調(diào)用了三級(jí)工廠中該對(duì)象的ObjectFactory方法,基于raw bean生成了對(duì)象放入到了二級(jí)緩存。但是當(dāng)raw bean執(zhí)行完initializeBean之后生成了新的對(duì)象,那就出問題了。如下圖:

在這里插入圖片描述

也就是說基于raw bean,得到了兩個(gè)基于該raw bean生成的proxy對(duì)象,Spring容器不知道最終該在容器里保存哪一個(gè)了。

2.2 循環(huán)依賴放入二級(jí)緩存處邏輯

1.每個(gè)bean在進(jìn)行屬性注入之前,默認(rèn)都會(huì)往Spring容器中放入一個(gè)ObjectFactory進(jìn)入三級(jí)工廠,以便自己在屬性注入的時(shí)候被循環(huán)依賴時(shí)調(diào)用生成對(duì)象

if (earlySingletonExposure) {
	// 返回一個(gè)進(jìn)行了aop處理的ObjectFactory,提前暴露
	// 但是只有當(dāng)該實(shí)例在創(chuàng)建過程中還被其他實(shí)例引用(循環(huán)依賴),才會(huì)被調(diào)用getEarlyBeanReference
	// 此處是第四次調(diào)用beanPostProcessor,不一定會(huì)調(diào)用,只有當(dāng)該類真的在創(chuàng)建過程中被其他類當(dāng)做屬性所依賴
	addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}

2.所以在創(chuàng)建CycleService1過程中,CycleService2去注入CycleService2之前在三級(jí)工廠里放入了自己的ObjectFactory對(duì)象,然后在CycleService2創(chuàng)建過程中,要注入CycleService1的時(shí)候,就會(huì)調(diào)用Spring容器中的getEarlyBeanReference(beanName, mbd, bean)獲取CycleService1,下面我們來看看該方法調(diào)用的具體邏輯

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;
					exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName);
				}
			}
		}
		return exposedObject;
	}

在這里插入圖片描述

然后咱們debug發(fā)現(xiàn),只有AbstractAutoProxyCreator#getEarlyBeanReference方法,有具體實(shí)現(xiàn)邏輯

public Object getEarlyBeanReference(Object bean, String beanName) {
		Object cacheKey = getCacheKey(bean.getClass(), beanName);
		this.earlyProxyReferences.put(cacheKey, bean);
		return wrapIfNecessary(bean, beanName, cacheKey);
	}

具體的細(xì)節(jié),我們就不進(jìn)入的,主要就是通過調(diào)用wrapIfNecessary生成了raw bean的aop proxy bean,后面放入了二級(jí)緩存。

2.3 initializeBean生成的對(duì)象

在initializeBean方法里會(huì)調(diào)用applyBeanPostProcessorsAfterInitialization方法

@Override
	public Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName)
			throws BeansException {

		Object result = existingBean;
		for (BeanPostProcessor processor : getBeanPostProcessors()) {
			Object current = processor.postProcessAfterInitialization(result, beanName);
			if (current == null) {
				return result;
			}
			result = current;
		}
		return result;
	}

在循環(huán)里面會(huì)調(diào)用postProcessAfterInitialization方法
重點(diǎn)關(guān)注AbstractAutoProxyCreator的該方法實(shí)現(xiàn):

@Override
	public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {
		if (bean != null) {
			Object cacheKey = getCacheKey(bean.getClass(), beanName);
			if (this.earlyProxyReferences.remove(cacheKey) != bean) {
				// 對(duì)bean進(jìn)行proxy操作
				return wrapIfNecessary(bean, beanName, cacheKey);
			}
		}
		return bean;
	}

會(huì)發(fā)現(xiàn)AbstractAutoProxyCreator#postProcessAfterInitialization里面的具體邏輯就是判斷這個(gè)類有沒有調(diào)用過wrapIfNecessary,如果調(diào)用過就不再調(diào)用,就是保證同一個(gè)raw bean不會(huì)被多次proxy,同時(shí)提前暴露注入到其他對(duì)象里的就是proxy bean。
但是由于該bean(CycleService1)上加了@Async注解,此次也會(huì)觸發(fā)AsyncAnnotationBeanPostProcessor#postProcessAfterInitialization,而這個(gè)方法,我們?cè)?a href="http://www.dbjr.com.cn/article/221542.htm" target="_blank">這篇文章里講過了,正是@Async注解能生效的關(guān)鍵邏輯。所以此處生成了一個(gè)具有Async功能的新的async proxy bean

2.4 再次分析原因

基于2.3和2.4,我們基于raw bean得到了二級(jí)緩存里的aop proxy bean和async proxy bean。
讓我們?cè)倩貞浺幌屡袛噙壿嫞?/p>

//此處是從二級(jí)緩存里面根據(jù)beanName拿出對(duì)象,因?yàn)槎?jí)緩存里放入的是因?yàn)檠h(huán)依賴給其他bean注入的代理對(duì)象
	Object earlySingletonReference = getSingleton(beanName, false);
	if (earlySingletonReference != null) {
		if (exposedObject == bean) {
			exposedObject = earlySingletonReference;
		}
		// 我們之前早期暴露出去的Bean跟現(xiàn)在最后要放到容器中的Bean不是同一個(gè)
		// allowRawInjectionDespiteWrapping為false
		// 并且當(dāng)前Bean被當(dāng)成依賴注入到了別的Bean中
		else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {
			// 獲取到當(dāng)前Bean依賴的Bean
			String[] dependentBeans = getDependentBeans(beanName);
			// 要得到真實(shí)的依賴的Bean
			Set<String> actualDependentBeans = new LinkedHashSet<>(dependentBeans.length);
			for (String dependentBean : dependentBeans) {
				// 移除那些僅僅為了類型檢查而創(chuàng)建出來
				if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {
					actualDependentBeans.add(dependentBean);
				}
			}
			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.");
			}
		}
	}

簡(jiǎn)而言之,也就是此時(shí)從二級(jí)緩存里拿到了aop proxy bean,同時(shí)了執(zhí)行完initializeBean之后,raw bean變?yōu)榱薬sync proxybean,Spring容器基于raw bean得到了兩個(gè)proxy bean,無法處理了。所以在使用@Async注解時(shí),盡量不要在被循環(huán)依賴的Class上添加

解決方案

打破循環(huán)依賴

目前我能想到的方法就是打破循環(huán)依賴,因?yàn)檠h(huán)依賴發(fā)生在bean生命周期的–屬性注入階段,所以我們需要做的就是打破這種循環(huán)依賴

1.延遲注入(使用@Lazy注解)

@Service
public class CycleService1 {

	@Lazy
	@Autowired
	private CycleService2 cycleService2;

	@WangAnno
	@Async
	public void doThings() {
		cycleService2.alsoDo();
		System.out.println("it's a async move");
	}

}

看過這篇文章的都知道原理了,此處不再累贅

2. 手動(dòng)延遲注入(使用applicationContext.getBean)

@Service
public class CycleService1 {

	@Autowired
	private ApplicationContext applicationContext;

	private CycleService2 cycleService2;

	@WangAnno
	@Async
	public void doThings() {
		if (Objects.isNull(cycleService2)) {
			cycleService2 = applicationContext.getBean(CycleService2.class);
		}
		cycleService2.alsoDo();
		System.out.println("it's a async move");
	}

}

其實(shí)效果是上面加了@Lazy效果是一樣的,不過是我們自己在方法執(zhí)行的過程中手動(dòng)進(jìn)行延遲注入而已。

總結(jié)

在這里插入圖片描述

從二級(jí)緩存里拿到earlySingletonReference(aop proxy bean),同時(shí)了執(zhí)行完initializeBean之后,raw bean變?yōu)榱薳xposedObject(async proxy bean),Spring容器基于raw bean得到了兩個(gè)proxy bean,無法處理了。
所以在使用@Async注解時(shí),盡量不要在被循環(huán)依賴的Class上添加
實(shí)在非要添加,可以看看我給出的解決方法。

到此這篇關(guān)于Java @Async注解導(dǎo)致spring啟動(dòng)失敗解決方案詳解的文章就介紹到這了,更多相關(guān)Java @Async注解導(dǎo)致spring啟動(dòng)失敗解決方案內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java實(shí)現(xiàn)視頻時(shí)間維度剪切的工具類

    Java實(shí)現(xiàn)視頻時(shí)間維度剪切的工具類

    這篇文章主要為大家詳細(xì)介紹了將視頻按照時(shí)間維度進(jìn)行剪切的Java工具類,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下
    2022-12-12
  • Java8新特性之Lambda表達(dá)式淺析

    Java8新特性之Lambda表達(dá)式淺析

    這篇文章主要介紹了Java8新特性之Lambda表達(dá)式,本文著重講解了Lambda表達(dá)式的語法部份,需要的朋友可以參考下
    2014-06-06
  • java使用鏈表實(shí)現(xiàn)約瑟夫環(huán)

    java使用鏈表實(shí)現(xiàn)約瑟夫環(huán)

    這篇文章主要為大家詳細(xì)介紹了java使用鏈表實(shí)現(xiàn)約瑟夫環(huán),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • 10張圖總結(jié)出并發(fā)編程最佳學(xué)習(xí)路線

    10張圖總結(jié)出并發(fā)編程最佳學(xué)習(xí)路線

    這篇文章主要介紹了并發(fā)編程的最佳學(xué)習(xí)路線,文中通過圖片介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-08-08
  • 分析那些不講武德的SDK(構(gòu)造使用規(guī)范)

    分析那些不講武德的SDK(構(gòu)造使用規(guī)范)

    這篇文章主要為大家介紹了盤點(diǎn)分析那些不講武德的SDK(構(gòu)造規(guī)范)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • Java中invokedynamic字節(jié)碼指令問題

    Java中invokedynamic字節(jié)碼指令問題

    這篇文章主要介紹了Java中invokedynamic字節(jié)碼指令問題,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2019-04-04
  • Springboot2.6.x的啟動(dòng)流程與自動(dòng)配置詳解

    Springboot2.6.x的啟動(dòng)流程與自動(dòng)配置詳解

    這篇文章主要給大家介紹了關(guān)于Springboot2.6.x的啟動(dòng)流程與自動(dòng)配置的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2022-01-01
  • java中的stream流中的并行查詢java

    java中的stream流中的并行查詢java

    Stream流是Java 8引入的一種新的數(shù)據(jù)處理方式,它提供了一種高效、簡(jiǎn)潔的數(shù)據(jù)處理方式,Stream流可以讓我們以聲明式的方式處理數(shù)據(jù),提高了代碼的可讀性和可維護(hù)性,同時(shí),Stream流支持并行處理,可以充分利用多核CPU的性能,提高程序的運(yùn)行效率
    2024-07-07
  • springboot微服務(wù)項(xiàng)目集成html頁(yè)面的實(shí)現(xiàn)

    springboot微服務(wù)項(xiàng)目集成html頁(yè)面的實(shí)現(xiàn)

    本文主要介紹了springboot微服務(wù)項(xiàng)目集成html頁(yè)面的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • 如何使用stream從List對(duì)象中獲取某列數(shù)據(jù)

    如何使用stream從List對(duì)象中獲取某列數(shù)據(jù)

    這篇文章主要介紹了如何使用stream從List對(duì)象中獲取某列數(shù)據(jù)問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-12-12

最新評(píng)論