Spring AOP無(wú)法攔截內(nèi)部方法調(diào)用

假設(shè)一個(gè)接口里面有兩個(gè)方法:

package demo.long;

public interface CustomerService {  
    public void doSomething1();  
    public void doSomething2();  
}  

接口實(shí)現(xiàn)類如下:

package demo.long.impl;

import demo.long.CustomerService; 

public class CustomerServiceImpl implements CustomerService {  
  
    public void doSomething1() {  
        System.out.println("CustomerServiceImpl.doSomething1()");  
        doSomething2();  
    }  
  
    public void doSomething2() {  
        System.out.println("CustomerServiceImpl.doSomething2()");  
    }  
  
}  

現(xiàn)在我需要在CustomerService接口的每個(gè)方法被調(diào)用時(shí)都在方法前執(zhí)行一些邏輯脆粥,所以需要配置一個(gè)攔截器:

package demo.long;

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;

@Aspect
public class CustomerServiceInterceptor {

    @Before("execution(* demo.long..*.*(..))")
    public void doBefore() {
        System.out.println("do some important things before..."); 
    }
}

把Bean加到Spring配置中

<aop:aspectj-autoproxy />

<bean id="customerService" class="demo.long.impl.CustomerServiceImpl" />
<bean id="customerServiceInterceptor" class="demo.long.CustomerServiceInterceptor" />

如果現(xiàn)在外部對(duì)象調(diào)用CustomerService的doSomething1()方法的時(shí)候,會(huì)發(fā)現(xiàn)只有doSomething1()方法執(zhí)行前打印了“do some important things before...”,而doSomething1()內(nèi)部調(diào)用doSomething2()時(shí)并沒(méi)有打印上述內(nèi)容;外部對(duì)象單獨(dú)調(diào)用doSomething2()時(shí)會(huì)打印上述內(nèi)容。

public class CustomerServiceTest {

    @Autowired
    ICustomerService customerService;

    @Test
    public void testAOP() {
        customerService.doSomething1();
    }
}

原因分析

攔截器的實(shí)現(xiàn)原理就是動(dòng)態(tài)代理包个,實(shí)現(xiàn)AOP機(jī)制。Spring 的代理實(shí)現(xiàn)有兩種:一是基于 JDK Dynamic Proxy 技術(shù)而實(shí)現(xiàn)的冤留;二是基于 CGLIB 技術(shù)而實(shí)現(xiàn)的碧囊。如果目標(biāo)對(duì)象實(shí)現(xiàn)了接口,在默認(rèn)情況下Spring會(huì)采用JDK的動(dòng)態(tài)代理實(shí)現(xiàn)AOP纤怒,CustomerServerImpl正是這種情況糯而。

JDK動(dòng)態(tài)代理生成的CustomerServiceImpl的代理類大致如下:

public class CustomerServiceProxy implements CustomerService {  
  
    private CustomerService customerService;  
  
    public void setCustomerService(CustomerService customerService) {  
        this.customerService = customerService;  
    }  
  
    public void doSomething1() {  
        doBefore();  
        customerService.doSomething1();  
    }  
  
    public void doSomething2() {  
        doBefore();  
        customerService.doSomething2();  
    }  
  
    private void doBefore() {  
        // 例如,可以在此處開(kāi)啟事務(wù)或記錄日志
        System.out.println("do some important things before...");  
    }  
  
}  

客戶端程序使用代理類對(duì)象去調(diào)用業(yè)務(wù)邏輯:

public class TestProxy {  
      
    public static void main(String[] args) {  
        // 創(chuàng)建代理目標(biāo)對(duì)象
        // 對(duì)于Spring來(lái)說(shuō)泊窘,這一工作是由Spring容器完成的熄驼。  
        CustomerService serviceProxyTarget = new CustomerServiceImpl();  
  
        // 創(chuàng)建代理對(duì)象
        // 對(duì)于Spring來(lái)說(shuō),這一工作也是由Spring容器完成的烘豹。 
        CustomerServiceProxy serviceProxy = new CustomerServiceProxy();  
        serviceProxy.setCustomerService(serviceProxyTarget);  
        CustomerService serviceBean = (CustomerService) serviceProxy;  
  
        // 調(diào)用業(yè)務(wù)邏輯操作  
        serviceBean.doSomething1();  
    }  
}  

執(zhí)行main方法瓜贾,發(fā)現(xiàn)doSomething1()中調(diào)用doSomething2()方法的時(shí)候并未去執(zhí)行CustomerServiceProxy類的doBefore()方法。其實(shí)doSomething2()等同于this.doSomething2()携悯,在CustomerServiceImpl類中this關(guān)鍵字表示的是當(dāng)前這個(gè)CustomerServiceImpl類的實(shí)例祭芦,所以程序會(huì)去執(zhí)行CustomerServiceImpl對(duì)象中的doSomething2()方法,而不會(huì)去執(zhí)行CustomerServiceProxy類對(duì)象中的 doSomething2()方法憔鬼。

在使用Spring AOP的時(shí)候龟劲,我們從IOC容器中獲取的Bean對(duì)象其實(shí)都是代理對(duì)象胃夏,而不是那些Bean對(duì)象本身,由于this關(guān)鍵字引用的并不是該Service Bean對(duì)象的代理對(duì)象昌跌,而是其本身仰禀,因此Spring AOP是不能攔截到這些被嵌套調(diào)用的方法的。

解決方案

  1. 修改類蚕愤,不要出現(xiàn)“自調(diào)用”的情況:這是Spring文檔中推薦的“最佳”方案悼瘾;
  2. 若一定要使用“自調(diào)用”,那么this.doSomething2()替換為:((CustomerService) AopContext.currentProxy()).doSomething2()审胸;此時(shí)需要修改spring的aop配置:
<aop:aspectj-autoproxy expose-proxy="true" />
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市卸勺,隨后出現(xiàn)的幾起案子砂沛,更是在濱河造成了極大的恐慌,老刑警劉巖曙求,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件碍庵,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡悟狱,警方通過(guò)查閱死者的電腦和手機(jī)静浴,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)挤渐,“玉大人苹享,你說(shuō)我怎么就攤上這事≡÷椋” “怎么了得问?”我有些...
    開(kāi)封第一講書人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)软免。 經(jīng)常有香客問(wèn)我宫纬,道長(zhǎng),這世上最難降的妖魔是什么膏萧? 我笑而不...
    開(kāi)封第一講書人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任漓骚,我火速辦了婚禮,結(jié)果婚禮上榛泛,老公的妹妹穿的比我還像新娘蝌蹂。我一直安慰自己,他們只是感情好曹锨,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布叉信。 她就那樣靜靜地躺著,像睡著了一般艘希。 火紅的嫁衣襯著肌膚如雪硼身。 梳的紋絲不亂的頭發(fā)上硅急,一...
    開(kāi)封第一講書人閱讀 49,166評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音佳遂,去河邊找鬼营袜。 笑死,一個(gè)胖子當(dāng)著我的面吹牛丑罪,可吹牛的內(nèi)容都是我干的荚板。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼吩屹,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼跪另!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起煤搜,我...
    開(kāi)封第一講書人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤免绿,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后擦盾,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體嘲驾,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年迹卢,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了辽故。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡腐碱,死狀恐怖誊垢,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情症见,我是刑警寧澤彤枢,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站筒饰,受9級(jí)特大地震影響缴啡,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜瓷们,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一业栅、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧谬晕,春花似錦碘裕、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春文兢,著一層夾襖步出監(jiān)牢的瞬間晤斩,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工姆坚, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留澳泵,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓兼呵,卻偏偏與公主長(zhǎng)得像兔辅,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子击喂,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理维苔,服務(wù)發(fā)現(xiàn),斷路器懂昂,智...
    卡卡羅2017閱讀 134,601評(píng)論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,748評(píng)論 6 342
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法介时,類相關(guān)的語(yǔ)法,內(nèi)部類的語(yǔ)法忍法,繼承相關(guān)的語(yǔ)法,異常的語(yǔ)法榕吼,線程的語(yǔ)...
    子非魚_t_閱讀 31,587評(píng)論 18 399
  • 微涼的夜風(fēng)輕輕拂過(guò) 點(diǎn)點(diǎn)星光眨著眼眸饿序, 把心中的想念裝在船上 披一身如水的月光 我想劃著那彎月亮去看你 輕輕靠近你...
    蘭馨若冰閱讀 212評(píng)論 9 3
  • 今年的氣溫回升特別慢原探,不管清晨跑步還是傍晚,天都是黑黑的顽素。 從開(kāi)始跑步咽弦,到現(xiàn)在已經(jīng)第二個(gè)年頭了。和以前的自己相比胁出,...
    天使也掉毛毛閱讀 1,374評(píng)論 10 15