2021-02-03-aop無法切到service內(nèi)部調(diào)用的方法

轉(zhuǎn)至鏈接:https://blog.csdn.net/qq_37372909/article/details/79979712

一。場景描述

有一個控制層類OutStoreOverController(簡稱controller),依賴了XsCustomorExpenseOperateServiceImpl(簡稱service)類。controller在2個不同方法中分別調(diào)用了service的siteDeliverySettlement 和stockDownAccounts方法(分別簡稱為m1和m2)。m1和m2在具體實現(xiàn)的時候又調(diào)用了service的內(nèi)部方法createExpense(申明為public,簡稱為m3)

方法調(diào)用的時序圖如下:


image.png

現(xiàn)在有一個切面StorageOperateOMSAopServiceImpl,需要切createExpense(m3)方法,在m3方法執(zhí)行前做點事情。經(jīng)過配置后,運行發(fā)現(xiàn)m3方法并沒有被切到。

二。問題分析

當(dāng)controller構(gòu)建實例的時候,注入service實例的時候,發(fā)現(xiàn)其有切面,產(chǎn)生了代理類serviceProxy并注入給了controller。

實際調(diào)用的時序圖如下:

image

這樣就導(dǎo)致m3方法根本沒有被切面切入。雖然controller第一次調(diào)用的是代理類,但是在調(diào)用m3方法的時候是調(diào)用的service實例內(nèi)部的m3方法,所以切面沒有生效。

三。問題解決

原來m1、m2方法調(diào)用m3方法時為:

XXXXXXXXXX;

m3();

XXXXXXXX;

修改后的寫法為:

XXXXXXXX;

Service serviceTemp=ApplicationContextUtil.getBean(Service.class);

serviceTemp.m3();

XXXXXX;

修改后調(diào)用的時序圖為:

image.png

真正使切面生效的就是

Service serviceTemp=ApplicationContextUtil.getBean(Service.class);

這一行代碼。向spring容器拿的實例,實際上是代理類servciceProxy。調(diào)用代理類的m3方法就會去先執(zhí)行aop中前置切面代碼,再會調(diào)用真正service實例的m3方法。最終,aop才有效果了。需要理解基于動態(tài)代理的aop原理。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 京東二面后整理的spring面試題 Q:ioc是什么,有什么用? A:Ioc全稱Inversion of Cont...
    Java__JJ閱讀 445評論 0 0
  • 目錄 1 數(shù)據(jù)交換格式 2 Java反射 3 Spring簡介 4 Spring IOC 5 代理 6...
    小小千千閱讀 444評論 1 0
  • 一. Java基礎(chǔ)部分.................................................
    wy_sure閱讀 4,033評論 0 11
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,834評論 18 399
  • 多態(tài) 任何域的訪問操作都將有編譯器解析,如果某個方法是靜態(tài)的,它的行為就不具有多態(tài)性 java默認對象的銷毀順序與...
    yueyue_projects閱讀 1,095評論 0 1

友情鏈接更多精彩內(nèi)容