Spring?AOP?創(chuàng)建代理對象詳情
1 前言
在這篇文章中中已經(jīng)講述了 AOP 的后置處理器和增強(qiáng)信息的獲取,在本文中將繼續(xù)分享 AOP 創(chuàng)建代理對象和上下文相關(guān)的內(nèi)容。
2 創(chuàng)建代理對象
Spring AOP 使用 JDKProxy 和 CGLIB 兩種方式來生成代理對象,具體使用哪一種需要根據(jù) AopProxyFactory 接口的 createProxy 方法中的 AdvisedSupport 中的參數(shù)進(jìn)行確定,默認(rèn)情況下如果目標(biāo)類是接口,則使用 jdk 動(dòng)態(tài)代理技術(shù),如果是非接口類,則使用 cglib 來生成代理。
在獲取了所有對應(yīng) bean 的增強(qiáng)器之后,便可以進(jìn)行代理的創(chuàng)建了,createProxy 方法。 對于代理類的創(chuàng)建以及處理,spring 委托給 ProxyFactory 進(jìn)行處理,此方法主要進(jìn)行初始化操作,并為代理工作做準(zhǔn)備。主要分為以下步驟:
- 1 獲取當(dāng)前代理類的屬性。
- 2 添加代理接口,封裝
Advisor并加入ProxyFactory代理工廠中。 - 3 設(shè)置要代理的類,
spring還添加了customizeProxyFactory,子類可以在改方法中對ProxyFactory進(jìn)行包裝。 - 4 進(jìn)行獲取代理類的操作。

3 AOPContext Aop 上下文
在最終的代碼調(diào)用中,還是 JdkDynamicAopProxy 的 invoke 方法和 CglibAopProxy 的 intercept 方法,以 CglibAopProxy 為例,可以看到 AopContext.setCurrentProxy 調(diào)用,將當(dāng)前的對象設(shè)置到上下文中,在最后 finally 代碼塊中會(huì)將當(dāng)前代理對象移除。

AOPContext 的主要方法如下圖所示:

4 AOP 分析匯總
綜上的逐步分析,最終總結(jié)的調(diào)用鏈路圖如下所示,分為注冊 Bean, 尋找增強(qiáng)器和創(chuàng)建代理等主要環(huán)節(jié):
# 注冊 bean invokeBeanFactoryPostProcessors @EnableAspectJAutoProxy AspectJAutoProxyRegistrar.registerBeanDefinitions AopConfigUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source); AopConfigUtils.registerOrEscalateApcAsRequired # 調(diào)用 finishBeanFactoryInitialization AbstractAutoProxyCreator.postProcessAfterInitialization AbstractAutoProxyCreator.wrapIfNecessary AbstractAutoProxyCreator.getAdvicesAndAdvisorsForBean AbstractAdvisorAutoProxyCreator.getAdvicesAndAdvisorsForBean AbstractAdvisorAutoProxyCreator.findEligibleAdvisors AbstractAdvisorAutoProxyCreator.findCandidateAdvisors(); AnnotationAwareAspectJAutoProxyCreator.findCandidateAdvisors BeanFactoryAspectJAdvisorsBuilder.buildAspectJAdvisors ReflectiveAspectJAdvisorFactory.getAdvisors ReflectiveAspectJAdvisorFactory.getPointcut 獲取切點(diǎn)的注解信息 new InstantiationModelAwarePointcutAdvisorImpl instantiateAdvice ReflectiveAspectJAdvisorFactory.getAdvice 獲取 advice switch AspectJAroundAdvice AbstractAspectJAdvice.invokeAdviceMethod AspectJMethodBeforeAdvice AspectJAfterAdvice AspectJAfterReturningAdvice AspectJAfterThrowingAdvice AbstractAdvisorAutoProxyCreator.findAdvisorsThatCanApply(candidateAdvisors, beanClass, beanName); AbstractAutoProxyCreator.createProxy AbstractAutoProxyCreator.buildAdvisors proxyFactory.getProxy(classLoader) DefaultAopProxyFactory.createAopProxy JdkDynamicAopProxy.getProxy ObjenesisCglibAopProxy.getProxy
5 JDK 動(dòng)態(tài)代理和 CGLIB 的區(qū)別與聯(lián)系
在 Spring 中,如果說 IOC 和 DI 解決了類與類之間的耦合,那么動(dòng)態(tài)代理則是解決了方法與方法,業(yè)務(wù)方法和切面邏輯之間的耦合。
JDK 動(dòng)態(tài)代理只能對實(shí)現(xiàn)了接口類生成代理,而不能針對類進(jìn)行代理。Cglib 是針對類進(jìn)行代理,針對指定的類生成子類,并覆蓋方法并進(jìn)行增強(qiáng),因?yàn)椴捎玫氖抢^承方式,所代理的類和方法不能是 final 修飾(final 的類或方法,是無法繼承的)。
jdk 動(dòng)態(tài)代理:創(chuàng)建一個(gè)實(shí)現(xiàn) InvocationHandler 接口的攔截器,使用 Proxy.newProxyInstance() 反射機(jī)制生成一個(gè)代理對象,在具體調(diào)用方法之前先調(diào)用 InvokeHandler.invoke 來處理。
Cglib 動(dòng)態(tài)代理:利用 ASM 字節(jié)碼增強(qiáng)框架,對代理對象生成的 class 文件加載進(jìn)來,通過修改字節(jié)碼生成子類進(jìn)行代理。
jdk 和 cglib 使用方法的區(qū)別:
- 1 目標(biāo)對象如果實(shí)現(xiàn)了接口,則默認(rèn)使用 jdk 動(dòng)態(tài)代理。
- 2 如果目標(biāo)對象實(shí)現(xiàn)了接口,可以強(qiáng)制使用 cglib。
- 3 如果目標(biāo)對象沒有實(shí)現(xiàn)接口,則必須使用 cglib, spring 會(huì)自動(dòng)在 jdk 和 cglib 代理之間進(jìn)行切換。
綜上, jdk 代理只能針對實(shí)現(xiàn)接口的類生成代理,而不能針對類。cglib 只能針對類進(jìn)行代理。
6 總結(jié)
在本文中主要講述了后置處理器和 AOP 的實(shí)現(xiàn)原理,以及 jdk 動(dòng)態(tài)代理和 cglib 代理之間的區(qū)別,主要涉及增強(qiáng)獲取和代理類的獲取,以及后置處理器的理解。
到此這篇關(guān)于Spring AOP 創(chuàng)建代理對象詳情的文章就介紹到這了,更多相關(guān)Spring AOP 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis-plus和Mybatis出現(xiàn)版本不兼容的問題解決
MyBatis-Plus?與?MyBatis?之間的兼容性問題通常是由于版本不匹配引起的,本文主要介紹了Mybatis-plus和Mybatis出現(xiàn)版本不兼容的問題解決,具有一定的參考價(jià)值,感興趣的可以了解一下2024-08-08
Java C++實(shí)現(xiàn)相同MD5加密算法的方式
這篇文章主要介紹了Java與C++實(shí)現(xiàn)相同MD5加密算法的方法,需要的朋友可以參考下面文章內(nèi)容2021-09-09
Java日期時(shí)間及日期相互轉(zhuǎn)換實(shí)現(xiàn)代碼
這篇文章主要介紹了Java日期時(shí)間及日期相互轉(zhuǎn)換實(shí)現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
使用JPA自定義VO類型轉(zhuǎn)換(EntityUtils工具類)
這篇文章主要介紹了使用JPA自定義VO類型轉(zhuǎn)換(EntityUtils工具類),具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
解決java.sql.SQLException:索引中丟失 IN或OUT 參數(shù)::x問題
這篇文章主要介紹了解決java.sql.SQLException:索引中丟失 IN或OUT 參數(shù)::x問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12
一起來學(xué)習(xí)Java IO的轉(zhuǎn)化流
這篇文章主要為大家詳細(xì)介紹了Java IO的轉(zhuǎn)化流,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-03-03

