解決SpringAop內(nèi)部調(diào)用時(shí)不經(jīng)過(guò)代理類的問(wèn)題
SpringAop
AOP代理織入時(shí)期
- 編譯時(shí)織入 - aspectj框架
- 類加載時(shí)織入 - aspectj框架
- 運(yùn)行時(shí)織入 - spring-aop
動(dòng)態(tài)代理方式
- JDK - 被代理對(duì)象必須需要實(shí)現(xiàn)接口
- CGLIB - 采用繼承被代理對(duì)象方式實(shí)現(xiàn)代理功能
解決SpringAop內(nèi)部調(diào)用時(shí)不經(jīng)過(guò)代理類(而是通過(guò)this)
方案一
通過(guò)編譯時(shí)織入或者類加載時(shí)織入代碼
方案二
通過(guò)當(dāng)前代理類調(diào)用目標(biāo)方法
getOne()方法中直接調(diào)用getAll()時(shí)是通過(guò)this對(duì)象,這時(shí)候getAll()方法上的@AopLog就不會(huì)被AOP掃描到
@Service
@Slf4j
public class ServiceImpl implements IService {
? ? @Override
? ? @AopLog("value=getOne")
? ? public void getOne() {
? ? ? ? log.info("getOne running");
? ? ? ? // 直接調(diào)用getAll()方法 = this.getAll()
? ? ? ? getAll();
? ? }
? ? @Override
? ? @AopLog("value=getAll")
? ? public void getAll() {
? ? ? ? log.info("getAll running");
? ? }
}修改getOne()方法,通過(guò)AopContext.currentProxy()方法獲取當(dāng)前代理類,通過(guò)代理類來(lái)調(diào)用getAll()方法,這時(shí)候就是通過(guò)代理類調(diào)用的
@Service
@Slf4j
public class ServiceImpl implements IService {
? ? @Override
? ? @AopLog("value=getOne")
? ? public void getOne() {
? ? ? ? log.info("getOne running");
? ? ? ? // 獲取當(dāng)前代理類,通過(guò)代理類來(lái)調(diào)用getAll()方法
? ? ? ? ((IService) AopContext.currentProxy()).getAll();
? ? }
? ? @Override
? ? @AopLog("value=getAll")
? ? public void getAll() {
? ? ? ? log.info("getAll running");
? ? }
? ? @AopLog("value=getById")
? ? private void getById() {
? ? ? ? log.info("getById running");
? ? }
}this使得SpringAop失效之謎
問(wèn)題描述
類Demo被AOP掃描到,其中有A和B兩個(gè)方法,A方法中調(diào)用了B方法,執(zhí)行A方法時(shí),B方法的代理沒(méi)有生效
問(wèn)題剖析
我們知道AOP底層使用JDK動(dòng)態(tài)代理和cglib動(dòng)態(tài)代理想結(jié)合,通過(guò)判斷去創(chuàng)建對(duì)應(yīng)的代理對(duì)象。
而不管是那種方法,最終執(zhí)行完代理后,都會(huì)執(zhí)行目標(biāo)方法:method.invoke(target,agrs)-->傳入目標(biāo)對(duì)象
所以執(zhí)行A方法的代理后,執(zhí)行A的目標(biāo)方法,此時(shí)執(zhí)行的對(duì)象是目標(biāo)對(duì)象,所以目標(biāo)對(duì)象執(zhí)行A方法是A中隱藏的this
指的就是目標(biāo)對(duì)象,即執(zhí)行B方法的不再是代理對(duì)象而是目標(biāo)對(duì)象,故B方法不會(huì)被代理
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
詳解java中動(dòng)態(tài)代理實(shí)現(xiàn)機(jī)制
這篇文章主要為大家介紹了java中動(dòng)態(tài)代理實(shí)現(xiàn)機(jī)制的相關(guān)資料,需要的朋友可以參考下2016-01-01
Maven搭建springboot項(xiàng)目的方法步驟
這篇文章主要介紹了Maven搭建springboot項(xiàng)目的方法步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
深入學(xué)習(xí)Java單元測(cè)試(Junit+Mock+代碼覆蓋率)
在做單元測(cè)試時(shí),代碼覆蓋率常常被拿來(lái)作為衡量測(cè)試好壞的指標(biāo),甚至,用代碼覆蓋率來(lái)考核測(cè)試任務(wù)完成情況,比如,代碼覆蓋率必須達(dá)到80%或 90%。下面我們就來(lái)詳細(xì)學(xué)習(xí)下java單元測(cè)試吧2019-06-06
Mybatis-plus如何通過(guò)反射實(shí)現(xiàn)動(dòng)態(tài)排序不同字段功能
這篇文章主要介紹了Mybatis-plus如何通過(guò)反射實(shí)現(xiàn)動(dòng)態(tài)排序不同字段功能,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02
Java中LambdaQueryWrapper的常用方法詳解
這篇文章主要給大家介紹了關(guān)于Java中LambdaQueryWrapper常用方法的相關(guān)資料,lambdaquerywrapper是一個(gè)Java庫(kù),用于構(gòu)建類型安全的Lambda表達(dá)式查詢,需要的朋友可以參考下2023-11-11
Java發(fā)送http請(qǐng)求的示例(get與post方法請(qǐng)求)
這篇文章主要介紹了Java發(fā)送http請(qǐng)求的示例(get與post方法請(qǐng)求),幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2021-01-01
SpringBoot之使用Redis實(shí)現(xiàn)分布式鎖(秒殺系統(tǒng))
這篇文章主要介紹了SpringBoot之使用Redis實(shí)現(xiàn)分布式鎖(秒殺系統(tǒng)),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
ANSI,Unicode,BMP,UTF等編碼概念實(shí)例講解
這篇文章主要介紹了ANSI,Unicode,BMP,UTF等編碼概念實(shí)例講解,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-12-12
Springboot集成fastDFS配置過(guò)程解析
這篇文章主要介紹了Springboot集成fastDFS配置過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11

