AOP失效了...分析

技術(shù)來自于需求
組內(nèi)在做性能測試,要知道每個方法調(diào)用時候,外層方法以及內(nèi)部調(diào)用的每個子方法的耗時時長,第一時間呢就想到了在方法前后打印時間,然后做差值打印
但是里面的調(diào)用鏈比較長,然后寫了好多好多差值打印代碼...

基于上述情況呢,我就想著做個annotation給方法去使用,想打印方法執(zhí)行時間的方法呢就把我的annotation加上去就行了
先說下額外的一個小選型廢棄

  • 摒棄threadlocal記錄時間,本來想直接用threadlocal記錄,然后在全局返回體內(nèi)攔截的適合去除時間即可,但是由于我們方法內(nèi)許多方法都是可以異步進行提高性能的,用到了多線程,所以這個方案就廢棄了
  • 最終方案呢 是用annotation增強我們的方法,將方法執(zhí)行時間打印到mdc里,然后在全局?jǐn)r截器(一個對方法返回值再封裝的攔截器形如 m,d,e)里加了一個t (map結(jié)構(gòu)),將我們的mdc關(guān)于時間打印的都放進去了

AOP失效了啥情況?

同一類里調(diào)用別的AOP方法,寫下偽代碼,類似下面代碼

@PostMapping("/timeRecode")
    @TimeRecord
    public String timeRecode() throws InterruptedException {
        time1();
        TimeUnit.SECONDS.sleep(1);
        time2();
        return "ok";
    }

    @TimeRecord
    public void time1() throws InterruptedException {
        TimeUnit.SECONDS.sleep(1);
    }

    @TimeRecord
    public void time2() throws InterruptedException {
        TimeUnit.SECONDS.sleep(1);
    }

按照我的預(yù)期應(yīng)該是三個方法都被攔截到的,但是實驗證明只有timeRecode被攔截了,time1().time2()沒被攔截,分析下,兩者不同的是前者是外部方法,而后面兩者是內(nèi)部調(diào)用的.

原因分析:

Spring AOP采用代理的方式實現(xiàn)AOP亏吝,我們編寫的橫切邏輯被添加到動態(tài)生成的代理對象中,只要我們調(diào)用的是代理對象腮介,則可以保證調(diào)用的是被增強的代理方法置森。而在代理對象中斗埂,不管你的橫切邏輯是怎樣的,也不管你增加了多少層的橫切邏輯凫海,有一點可以確定的是呛凶,你終歸會調(diào)用目標(biāo)對象的同一方法來調(diào)用原始的業(yè)務(wù)邏輯。
如果目標(biāo)對象中的原始方法依賴于其他對象行贪,那么Spring會注入所依賴對象的代理對象漾稀,從而保證依賴的對象的橫切邏輯能夠被正衬O校織入。而一旦目標(biāo)對象調(diào)用的是自身的其他方法時崭捍,問題就來了尸折,這種情況下,目標(biāo)對象調(diào)用的并不是代理對象的方法殷蛇,故被調(diào)用的方法無法織入橫切邏輯翁授。

我們這里方法 A 被調(diào)用,是基于 AOP 生成的 代理對象 進行的調(diào)用晾咪;方法 B 調(diào)用方法 A ,是 this 目標(biāo)對象 直接調(diào)用贮配,并不是代理對象進行調(diào)用

解決方案

通過代理對象調(diào)用~

    1. 同一個類內(nèi)方法互相調(diào)用可以拿到Spring給我們創(chuàng)建的代理谍倦,用代理調(diào)用就可以解決,解決如下:
final TestController proxy = (TestController) AopContext.currentProxy();
        proxy.time2();
    1. 采用注入的方式把要調(diào)用的bean先注入進來,再通過bean去調(diào)用,這也適用于同一個類,其實我們也可以把自己的bean注入到自己方法內(nèi)使用.
//構(gòu)造器注入其他bean
private final CacheManager cacheManager;
//annotation注入其他bean
@Autowired
private  Temp temp;
@Service
public class SomeService {
    //注入自己
    @Autowired
    private SomeService self
}

ps:

另在查閱資料時候發(fā)現(xiàn)有人的aop失效是因為沒開啟cglib,這里也提一下啊,想要使用aop,要做以下配置

  • 啟動類上加@EnableAspectJAutoProxy(proxyTargetClass = true, exposeProxy = true),開啟cglib代理,為啥呢?
    Spring AOP通過動態(tài)代理實現(xiàn)的泪勒,不同場景下代理的支持

      1. 如果要切入的目標(biāo)對象實現(xiàn)了接口昼蛀,默認(rèn)情況下Spring會直接采用JDK的動態(tài)代理的方式去實現(xiàn)AOP
    
      2. 如果要切入的目標(biāo)對象實現(xiàn)了接口,也可以通過代碼控制強制要求Spring使用CGLIB代理的方式去實現(xiàn)AOP
    
      3. 如果要切入的目標(biāo)對象沒有實現(xiàn)任何接口圆存,那么此時Spring會自動采用CGLIB庫叼旋,它會根據(jù)實際切入目標(biāo)的情況,自動在JDK動態(tài)代理和CGLIB之間做切換沦辙,但是必須在項目中加入CGLIB需要的jar包夫植,否則當(dāng)Spring判定需要使用CGLIB的時候,而你沒有引入CGLIB的jar包油讯,此時就會報錯详民,而當(dāng)你引入CGLIB的jar包后,不需要額外做任何關(guān)于CGLIB配置陌兑,Spring直接就可以根據(jù)規(guī)則自動轉(zhuǎn)換代理模式沈跨。
    

其他失效

  • (1) 在一個類內(nèi)部調(diào)用時,被調(diào)用方法的 AOP 聲明將不起作用兔综。

  • (2) 對于基于接口動態(tài)代理的 AOP 事務(wù)增強來說饿凛,由于接口的方法都必然是 public ,這就要求實現(xiàn)類的實現(xiàn)方法也必須是 public 的(不能是 protected软驰、private 等)涧窒,同時不能使用 static 的修飾符。 所以碌宴,可以實施接口動態(tài)代理的方法只能是使用 public 或 public final 修飾符的方法杀狡,其他方法不可能被動態(tài)代理,相應(yīng)的也就不能實施 AOP 增強贰镣,換句話說呜象,即不能進行 Spring 事務(wù)增強了膳凝。

  • (3) 基于 CGLib 字節(jié)碼動態(tài)代理的方案是通過擴展被增強類,動態(tài)創(chuàng)建其子類的方式進行 AOP 增強植入的恭陡。
    由于使用 final蹬音、static、private 修飾符的方法都不能被子類覆蓋休玩,這些方法將無法實施 AOP 增強著淆。所以方法簽名必須特別注意這些修飾符的使用,以免使方法不小心成為事務(wù)管理的漏網(wǎng)之魚拴疤。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末永部,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子呐矾,更是在濱河造成了極大的恐慌苔埋,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蜒犯,死亡現(xiàn)場離奇詭異组橄,居然都是意外死亡,警方通過查閱死者的電腦和手機罚随,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進店門玉工,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人淘菩,你說我怎么就攤上這事遵班。” “怎么了瞄勾?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵费奸,是天一觀的道長。 經(jīng)常有香客問我进陡,道長愿阐,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任趾疚,我火速辦了婚禮缨历,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘糙麦。我一直安慰自己辛孵,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布赡磅。 她就那樣靜靜地躺著魄缚,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上冶匹,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天习劫,我揣著相機與錄音,去河邊找鬼嚼隘。 笑死诽里,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的飞蛹。 我是一名探鬼主播谤狡,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼卧檐!你這毒婦竟也來了墓懂?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤霉囚,失蹤者是張志新(化名)和其女友劉穎拒贱,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體佛嬉,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年闸天,在試婚紗的時候發(fā)現(xiàn)自己被綠了暖呕。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡苞氮,死狀恐怖湾揽,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情笼吟,我是刑警寧澤库物,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站贷帮,受9級特大地震影響戚揭,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜撵枢,卻給世界環(huán)境...
    茶點故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一民晒、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧锄禽,春花似錦潜必、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至宵晚,卻和暖如春垂攘,著一層夾襖步出監(jiān)牢的瞬間维雇,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工搜贤, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留谆沃,地道東北人。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓仪芒,卻偏偏與公主長得像唁影,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子掂名,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,901評論 2 345

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