Spring?循環(huán)依賴之AOP實現(xiàn)詳情
前言:
我們接著上一篇文章繼續(xù)往下看,首先看一下下面的例子,前面的兩個serviceA和serviceB不變,我們添加一個BeanPostProcessor:
@Component
public class MyPostProcessor implements BeanPostProcessor {
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
if (beanName.equals("serviceA")){
System.out.println("create new ServiceA");
return new ServiceA();
}
return bean;
}
}運行一下,結果報錯了:
Exception in thread "main" org.springframework.beans.factory.BeanCurrentlyInCreationException:
Error creating bean with name 'serviceA': Bean with name 'serviceA'
has been injected into other beans [serviceB] 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.
在分析錯誤之前,我們再梳理一下正常循環(huán)依賴的過程:
- 1、初始化原生對象serviceA,放入三級緩存
- 2、serviceA填充屬性,發(fā)現(xiàn)依賴serviceB,創(chuàng)建依賴對象
- 3、創(chuàng)建serviceB,填充屬性發(fā)現(xiàn)依賴serviceA,從三級緩存中找到填充
- 4、執(zhí)行serviceB的后置處理器和回調方法,放入單例池
- 5、執(zhí)行serviceA的后置處理器和回調方法,放入單例池
再回頭看上面的錯誤,大意為在循環(huán)依賴中我們給serviceB注入了serviceA,但是注入之后我們又在后置處理器中對serviceA進行了包裝,因此導致了serviceB中注入的和最后生成的serviceA不一致。
但是熟悉aop的同學應該知道,aop的底層也是利用后置處理器實現(xiàn)的啊,那么為什么aop就可以正常執(zhí)行呢?我們添加一個切面橫切serviceA的getServiceB方法:
@Component
@Aspect
public class MyAspect {
@Around("execution(* com.hydra.service.ServiceA.getServiceB())")
public void invoke(ProceedingJoinPoint pjp){
try{
System.out.println("execute aop around method");
pjp.proceed();
}catch (Throwable e){
e.printStackTrace();
}
}
}先不看運行結果,代碼可以正常執(zhí)行不出現(xiàn)異常,那么aop是怎么實現(xiàn)的呢?
前面的流程和不使用aop相同,我們運行到serviceB需要注入serviceA的地方,調用getSingleton方法從三級緩存中獲取serviceA存儲的singletonFactory,調用getEarlyBeanReference方法。在該方法中遍歷執(zhí)行SmartInstantiationAwareBeanPostProcessor后置處理器的getEarlyBeanReference方法:

看一下都有哪些類實現(xiàn)了這個方法:

在spring中,就是這個AbstractAutoProxyCreator負責實現(xiàn)了aop,進入getEarlyBeanReference方法:
public Object getEarlyBeanReference(Object bean, String beanName) throws BeansException {
//beanName
Object cacheKey = getCacheKey(bean.getClass(), beanName);
this.earlyProxyReferences.put(cacheKey, bean);
//產生代理對象
return wrapIfNecessary(bean, beanName, cacheKey);
}earlyProxyReferences 是一個Map,用于緩存bean的原始對象,也就是執(zhí)行aop之前的bean,非常重要,在后面還會用到這個Map:
Map<Object, Object> earlyProxyReferences = new ConcurrentHashMap<>(16);
記住下面這個wrapIfNecessary方法,它才是真正負責生成代理對象的方法:

上面首先解析并拿到所有的切面,調用createProxy方法創(chuàng)建代理對象并返回。然后回到getSingleton方法中,將serviceA加入二級緩存,并從三級緩存中移除掉。

可以看到,二級緩存中的serviceA已經是被cglib代理過的代理對象了,當然這時的serviceA還是沒有屬性值填充的。
那么這里又會有一個問題,我們之前講過,在填充完屬性后,會調用后置處理器中的方法,而這些方法都是基于原始對象的,而不是代理對象。

在前一篇文章中我們也講過,在initializeBean方法中會執(zhí)行后置處理器,并且正常情況下aop也是在這里完成的。那么我們就要面臨一個問題,如果避免重復執(zhí)行aop的過程。在initializeBean方法中:
if (mbd == null || !mbd.isSynthetic()) {
wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
}調用applyBeanPostProcessorsAfterInitialization,執(zhí)行所有后置處理器的after方法:

執(zhí)行AbstractAutoProxyCreator的postProcessAfterInitialization方法:
public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) throws BeansException {
if (bean != null) {
Object cacheKey = getCacheKey(bean.getClass(), beanName);
if (this.earlyProxyReferences.remove(cacheKey) != bean) {
return wrapIfNecessary(bean, beanName, cacheKey);
}
}
return bean;
}earlyProxyReferences 我們之前說過非常重要,它緩存了進行aop之前的原始對象,并且這里參數(shù)傳入的Object也是原始對象。因此在這里執(zhí)行remove操作的判斷語句返回false,不會執(zhí)行if中的語句,不會再執(zhí)行一遍aop的過程。
回過頭來再梳理一下,因為之前進行過循環(huán)依賴,所以提前執(zhí)行了AbstractAutoProxyCreator的getEarlyBeanReference方法,執(zhí)行了aop的過程,在earlyProxyReferences中緩存了原生對象。因此在循環(huán)依賴的情況下,等式成立,直接返回。而在沒有循環(huán)依賴的普通情況下,earlyProxyReferences執(zhí)行remove返回為null,等式不成立,正常執(zhí)行aop流程。
需要注意的是,這個方法中最終返回的還是原始對象,而不是aop后的代理對象。執(zhí)行到這一步,我們先看一下嵌套的狀態(tài):

對外暴露的serviceA是原始對象,依賴的serviceB已經被注入了。而serviceB中依賴的serviceA是代理對象,并且這個代理對象依賴的serviceB還沒有被注入。
向下執(zhí)行:

再次通過getSingleton獲取serviceA:

這次我們通過二級緩存就可以拿到之前經過aop的代理對象,因此不用找三級緩存直接返回這個代理對象,并最終把這個代理對象添加到一級緩存單例池中。
到這,我們對三級緩存的作用做一個總結:
- 1、
singletonObjects:單例池,緩存了經過完整生命周期的bean - 2、
earlySingletonObjects:緩存了提前曝光的原始對象,注意這里存的還不是bean,這里存的對象經過了aop的代理,但是沒有執(zhí)行屬性的填充以及后置處理器方法的執(zhí)行 - 3、
singletonFactories:緩存的是ObjectFactory,主要用來去生成原始對象進行了aop之后得到的代理對象。在每個bean的生成過程中,都會提前在這里緩存一個工廠。如果沒有出現(xiàn)循環(huán)依賴依賴這個bean,那么這個工廠不會起到作用,按照正常生命周期執(zhí)行,執(zhí)行完后直接把本bean放入一級緩存中。如果出現(xiàn)了循環(huán)依賴依賴了這個bean,沒有aop的情況下直接返回原始對象,有aop的情況下返回代理對象。
全部創(chuàng)建流程結束,看一下結果:

我們發(fā)現(xiàn),在生成的serviceA的cglib代理對象中,serviceB屬性值并沒有被填充,只有serviceB中serviceA的屬性填充成功了。

可以看到如果使用cglib,在代理對象的target中會包裹一個原始對象,而原始對象的屬性是被填充過的。
那么,如果不使用cglib代理,而使用jdk動態(tài)代理呢?我們對之前的代碼進行一下改造,添加兩個接口:
public interface IServiceA {
public IServiceB getServiceB();
}
public interface IServiceB {
public IServiceA getServiceA();
}改造兩個Service類:
@Component
public class ServiceA implements IServiceA{
@Autowired
private IServiceB serviceB;
public IServiceB getServiceB() {
System.out.println("get ServiceB");
return this.serviceB;
}
}
@Component
public class ServiceB implements IServiceB{
@Autowired
private IServiceA serviceA;
public IServiceA getServiceA() {
System.out.println("get ServiceA");
return serviceA;
}
}執(zhí)行結果:

看一下serviceA的詳細信息:

同樣也是在target中包裹了原生對象,并在原生對象中注入了serviceB的實例。
綜上兩種方法,可以看出在我們執(zhí)行serviceA的getServiceB方法時,都無法正常獲取到其bean對象,都會返回一個null值。那么如果非要直接獲得這個serviceB應該怎么辦呢?
我們可以通過反射的方式,先看cglib代理情況下:
ServiceA serviceA= (ServiceA) context.getBean("serviceA");
Field h = serviceA.getClass().getDeclaredField("CGLIB$CALLBACK_0");
h.setAccessible(true);
Object dynamicAdvisedInterceptor = h.get(serviceA);
Field advised = dynamicAdvisedInterceptor.getClass().getDeclaredField("advised");
advised.setAccessible(true);
Object target = ((AdvisedSupport)advised.get(dynamicAdvisedInterceptor)).getTargetSource().getTarget();
ServiceA serviceA1= (ServiceA) target;
System.out.println(serviceA1.getServiceB());再看看jdk動態(tài)代理情況下:
IServiceA serviceA = (IServiceA) context.getBean("serviceA");
Field h=serviceA.getClass().getSuperclass().getDeclaredField("h");
h.setAccessible(true);
AopProxy aopProxy = (AopProxy) h.get(serviceA);
Field advised = aopProxy.getClass().getDeclaredField("advised");
advised.setAccessible(true);
Object target = ((AdvisedSupport)advised.get(aopProxy)).getTargetSource().getTarget();
ServiceA serviceA1= (ServiceA) target;
System.out.println(serviceA1.getServiceB());執(zhí)行結果都能獲取到serviceB的實例:

對aop情況下的循環(huán)依賴進行一下總結:spring專門為了處理aop情況下的循環(huán)依賴提供了特殊的解決方案,但是不論是使用jdk動態(tài)代理還是cglib代理,都在代理對象的內部包裹了原始對象,在原始對象中才有依賴的屬性。此外,如果我們使用了后置處理器對bean進行包裝,循環(huán)依賴的問題還是不能解決的。
總結:
最后對本文的重點進行一下總結:
- 1、spring通過借助三級緩存完成了循環(huán)依賴的實現(xiàn),這個過程中要清楚三級緩存分別在什么場景下發(fā)揮了什么具體作用
- 2、產生aop情況下,調用后置處理器并將生成的代理對象提前曝光,并通過額外的一個緩存避免重復執(zhí)行aop
- 3、二級緩存和三級緩存只有在產生循環(huán)依賴的情況下,才會真正起到作用
- 4、此外,除去本文中提到的通過屬性的方式注入依賴的情況外,大家可能會好奇如果使用構造函數(shù)能否實現(xiàn)循環(huán)依賴,結果是不可以的。具體的調用過程這里不再多說,有興趣的同學可以自己再對照源碼進行一下梳理。
到此這篇關于Spring 循環(huán)依賴之AOP實現(xiàn)詳情的文章就介紹到這了,更多相關Spring 循環(huán)依賴 AOP 實現(xiàn)內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
SpringBoot?Security使用MySQL實現(xiàn)驗證與權限管理
安全管理是軟件系統(tǒng)必不可少的的功能。根據(jù)經典的“墨菲定律”——凡是可能,總會發(fā)生。如果系統(tǒng)存在安全隱患,最終必然會出現(xiàn)問題,這篇文章主要介紹了SpringBoot安全管理Spring?Security基本配置2022-11-11
Java之while與do-while循環(huán)的用法詳解
在上一篇文章中,給大家講解了循環(huán)的概念,并重點給大家講解了for循環(huán)的使用。但在Java中,除了for循環(huán)之外,還有while、do-while、foreach等循環(huán)形式。這篇文章給大家講解while循環(huán)的使用2023-05-05

