欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring AOP注解失效的坑及JDK動(dòng)態(tài)代理

 更新時(shí)間:2018年03月27日 16:23:11   作者:白色夜空  
這篇文章主要介紹了Spring AOP注解失效的坑及JDK動(dòng)態(tài)代理,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧

@Transactional @Async等注解不起作用

之前很多人在使用Spring中的@Transactional, @Async等注解時(shí),都多少碰到過(guò)注解不起作用的情況。

為什么會(huì)出現(xiàn)這些情況呢?因?yàn)檫@些注解的功能實(shí)際上都是Spring AOP實(shí)現(xiàn)的,而其實(shí)現(xiàn)原理是通過(guò)代理實(shí)現(xiàn)的。

JDK動(dòng)態(tài)代理

以一個(gè)簡(jiǎn)單的例子理解一下JDK動(dòng)態(tài)代理的基本原理:

//目標(biāo)類接口
public interface JDKProxyTestService {
  void run();
}

//目標(biāo)類
public class JDKProxyTestServiceImpl implements JDKProxyTestService {
  public void run(){
    System.out.println("do something...");
  }
}

//代理類
public class TestJDKProxy implements InvocationHandler {

  private Object targetObject; //代理目標(biāo)對(duì)象

  //構(gòu)造代理對(duì)象
  public Object newProxy(Object targetObject) {
    this.targetObject = targetObject;
    return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),
        targetObject.getClass().getInterfaces(), this);
  }

  //利用反射,在原邏輯上進(jìn)行邏輯增強(qiáng)
  public Object invoke(Object proxy, Method method, Object[] args)
      throws Throwable {
    //模擬事務(wù)開(kāi)始
    assumeBeginTransaction();
    //原執(zhí)行邏輯
    Object ret = method.invoke(targetObject, args);
    //模擬事務(wù)提交
    assumeCommitTransaction();
    return ret;
  }

  private void assumeBeginTransaction() {
    System.out.println("模擬事務(wù)開(kāi)始...");
  }

  private void assumeCommitTransaction() {
    System.out.println("模擬事務(wù)提交...");
  }
}

//測(cè)試
public class Test { 
  public static void main(String[] args) {
    TestJDKProxy jdkProxy = new TestJDKProxy();
    JDKProxyTestService proxy = (JDKProxyTestService) jdkProxy.newProxy(new JDKProxyTestServiceImpl());
    proxy.run();
  }
}

上面的例子應(yīng)該能夠清楚的解釋JDK動(dòng)態(tài)代理的原理了。它利用反射機(jī)制,生成了一個(gè)實(shí)現(xiàn)代理接口的匿名類,在調(diào)用具體方法前調(diào)用InvokeHandler來(lái)處理。我們通過(guò)代理類對(duì)象調(diào)用方法時(shí),實(shí)際上會(huì)先調(diào)用其invoke方法,里面再調(diào)用原方法。這樣我們可以在原方法邏輯的前后統(tǒng)一添加處理邏輯。

Spring還有一種動(dòng)態(tài)代理方式是CGLIB動(dòng)態(tài)代理。它是把代理對(duì)象類的class文件加載進(jìn)來(lái),通過(guò)修改其字節(jié)碼生成子類來(lái)處理。雖然處理方式不一樣,但是代理的思想都是一致的。

如果被代理的目標(biāo)對(duì)象實(shí)現(xiàn)了接口,那么Spring會(huì)默認(rèn)使用JDK動(dòng)態(tài)代理。所有該目標(biāo)類型實(shí)現(xiàn)的接口都將被代理。若該目標(biāo)對(duì)象沒(méi)有實(shí)現(xiàn)任何接口,則創(chuàng)建一個(gè)CGLIB代理。

Spring AOP注解失效及解決

基于以上對(duì)于動(dòng)態(tài)代理原理的分析,我們來(lái)看以下兩個(gè)常見(jiàn)的問(wèn)題:

同一個(gè)類中,方法A調(diào)用方法B(方法B上加有注解),注解無(wú)效

針對(duì)所有的Spring AOP注解,Spring在掃描bean的時(shí)候如果發(fā)現(xiàn)有此類注解,那么會(huì)動(dòng)態(tài)構(gòu)造一個(gè)代理對(duì)象。

如果你想要通過(guò)類X的對(duì)象直接調(diào)用其中帶注解的A方法,此注解是有效的。因?yàn)榇藭r(shí),Spring會(huì)判斷你將要調(diào)用的方法上存在AOP注解,那么會(huì)使用類X的代理對(duì)象調(diào)用A方法。

但是假設(shè)類X中的A方法會(huì)調(diào)用帶注解的B方法,而你依然想要通過(guò)類X對(duì)象調(diào)用A方法,那么B方法上的注解是無(wú)效的。因?yàn)榇藭r(shí)Spring判斷你調(diào)用的A并無(wú)注解,所以使用的還是原對(duì)象而非代理對(duì)象。接下來(lái)A再調(diào)用B時(shí),在原對(duì)象內(nèi)B方法的注解當(dāng)然無(wú)效了。

解決方法:

最簡(jiǎn)單的方式當(dāng)然是可以讓方法A和B沒(méi)有依賴,能夠直接通過(guò)類X的對(duì)象調(diào)用B方法。

但是很多時(shí)候可能我們的邏輯拆成這樣寫并不好,那么就還有一種方法:想辦法手動(dòng)拿到代理對(duì)象。

AopContext類有一個(gè)currentProxy()方法,能夠直接拿到當(dāng)前類的代理對(duì)象。那么以上的例子,就可以這樣解決:

// 在A方法內(nèi)部調(diào)用B方法
// 1.直接調(diào)用B,注解失效。
B()
// 2.拿到代理類對(duì)象,再調(diào)用B。
((X)AopContext.currentProxy()).B()

AOP注解方法里使用@Autowired對(duì)象為null

在之前的使用中,出現(xiàn)過(guò)在加上注解的方法中,使用其他注入的對(duì)象時(shí),發(fā)現(xiàn)對(duì)象并沒(méi)有被注入進(jìn)來(lái),為null。

最終發(fā)現(xiàn),導(dǎo)致這種情況的原因是因?yàn)榉椒閜rivate。因?yàn)镾pring不管使用的是JDK動(dòng)態(tài)代理還是CGLIB動(dòng)態(tài)代理,一個(gè)是針對(duì)實(shí)現(xiàn)接口的類,一個(gè)是通過(guò)子類實(shí)現(xiàn)。無(wú)論是接口還是父類,顯然都不能出現(xiàn)private方法,否則子類或?qū)崿F(xiàn)類都不能覆蓋到。

如果方法為private,那么在代理過(guò)程中,根本找不到這個(gè)方法,引起代理對(duì)象創(chuàng)建出現(xiàn)問(wèn)題,也導(dǎo)致了有的對(duì)象沒(méi)有注入進(jìn)去。

所以如果方法需要使用AOP注解,請(qǐng)把它設(shè)置為非private方法。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Spring AOP快速入門及開(kāi)發(fā)步驟

    Spring AOP快速入門及開(kāi)發(fā)步驟

    Spring AOP(面向切面編程)核心概念包括切面(Aspect)、連接點(diǎn)(JoinPoint)、切點(diǎn)(Pointcut)、通知(Advice)等,通過(guò)在不改變?cè)a的情況下,對(duì)方法進(jìn)行增強(qiáng),實(shí)現(xiàn)了代碼的解耦和功能擴(kuò)展,本文帶來(lái)大家掌握Spring 中 AOP 的開(kāi)發(fā)步驟,感興趣的朋友一起看看吧
    2024-10-10
  • java使用CompletableFuture分批處理任務(wù)實(shí)現(xiàn)

    java使用CompletableFuture分批處理任務(wù)實(shí)現(xiàn)

    本文主要介紹了java使用CompletableFuture分批處理任務(wù)實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-07-07
  • Java編程—在測(cè)試中考慮多態(tài)

    Java編程—在測(cè)試中考慮多態(tài)

    這篇文章主要介紹了Java編程—在測(cè)試中考慮多態(tài),具有一定參考價(jià)值,需要的朋友可以了解下。
    2017-11-11
  • Java文件讀寫詳解

    Java文件讀寫詳解

    在真實(shí)的應(yīng)用場(chǎng)景中,很多時(shí)候需要使用?Java?讀寫文件。比如說(shuō),讀取配置文件信息、讀取用戶輸入等。本篇文章將會(huì)詳細(xì)介紹?Java?文件讀寫的相關(guān)知識(shí),其中包括:讀取文件、寫入文件、復(fù)制文件和刪除文件等操作,需要的朋友可以參考下
    2023-05-05
  • Flutter ListView 上拉加載更多下拉刷新功能實(shí)現(xiàn)方法

    Flutter ListView 上拉加載更多下拉刷新功能實(shí)現(xiàn)方法

    這篇文章主要介紹了Flutter ListView 上拉加載更多下拉刷新功能實(shí)現(xiàn)方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2019-07-07
  • Java中的ArrayList.trimToSize()方法詳解

    Java中的ArrayList.trimToSize()方法詳解

    這篇文章主要介紹了Java中的ArrayList.trimToSize()方法詳解,前幾天看了Java?ArrayList,沒(méi)有明白trimToSize()這個(gè)方法是什么意思,所以看了一下源碼并且debug一下自己的一個(gè)例子,明白了其中的含義,需要的朋友可以參考下
    2023-11-11
  • Java的Struts2框架配合Ext JS處理JSON數(shù)據(jù)的使用示例

    Java的Struts2框架配合Ext JS處理JSON數(shù)據(jù)的使用示例

    這篇文章主要介紹了Java的Struts2框架配合Ext JS處理JSON數(shù)據(jù)的使用示例,包括將Ext JS中的JSON數(shù)據(jù)解析為列表的方法,需要的朋友可以參考下
    2016-03-03
  • idea中方法、注釋、導(dǎo)入類折疊或是展開(kāi)的設(shè)置方法

    idea中方法、注釋、導(dǎo)入類折疊或是展開(kāi)的設(shè)置方法

    這篇文章主要介紹了idea中方法、注釋、導(dǎo)入類折疊或是展開(kāi)的設(shè)置,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-04-04
  • 詳解MyBatis-Plus Wrapper條件構(gòu)造器查詢大全

    詳解MyBatis-Plus Wrapper條件構(gòu)造器查詢大全

    這篇文章主要介紹了詳解MyBatis-Plus Wrapper條件構(gòu)造器查詢大全,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-08-08
  • 一篇文章帶你了解JVM內(nèi)存模型

    一篇文章帶你了解JVM內(nèi)存模型

    本文講解了Java 內(nèi)存模型來(lái)看看解決可見(jiàn)性、有序性問(wèn)題的 Java 內(nèi)存模型(JMM),今天通過(guò)本文給大家介紹Java 內(nèi)存模型(JVM)的相關(guān)知識(shí),感興趣的朋友一起看看吧
    2021-09-09

最新評(píng)論