1. 程式人生 > 程式設計 >詳解spring中aop不生效的幾種解決辦法

詳解spring中aop不生效的幾種解決辦法

先看下這個問題的背景:假設有一個spring應用,開發人員希望自定義一個註解@Log,可以加到指定的方法上,實現自動記錄日誌(入參、出參、響應耗時這些)

package com.cnblogs.yjmyzz.springbootdemo.aspect;
 
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
 
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {
 
}

然後再寫一個Aspect來解析這個註解,對打了Log註解的方法進行增強處理 

package com.cnblogs.yjmyzz.springbootdemo.aspect;
 
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
 
import java.lang.reflect.Method;
 
@Component
@Aspect
public class LogAspect {
 
  @Pointcut("execution (* com.cnblogs.yjmyzz.springbootdemo.service..*.*(..))")
  public void logPointcut() {
 
  }
 
  @Around("logPointcut()")
  public void around(JoinPoint point) {
    String methodName = point.getSignature().getName();
    Object[] args = point.getArgs();
    Class<?>[] argTypes = new Class[point.getArgs().length];
    for (int i = 0; i < args.length; i++) {
      argTypes[i] = args[i].getClass();
    }
    Method method = null;
    try {
      method = point.getTarget().getClass().getMethod(methodName,argTypes);
    } catch (Exception e) {
      e.printStackTrace();
    }
    //獲取方法上的註解
    Log log = method.getAnnotation(Log.class);
    if (log != null) {
      //演示方法執行前,記錄一行日誌
      System.out.println("before:" + methodName);
    }
    try {
      //執行方法
      ((ProceedingJoinPoint) point).proceed();
    } catch (Throwable throwable) {
      throwable.printStackTrace();
    } finally {
      if (log != null) {
        //演示方法執行後,記錄一行日誌
        System.out.println("after:" + methodName);
      }
    }
  }
}

寫一個測試Service類:

package com.cnblogs.yjmyzz.springbootdemo.service;
 
import com.cnblogs.yjmyzz.springbootdemo.aspect.Log;
import org.springframework.stereotype.Component;
 
@Component
public class HelloService {
   
  @Log
  public void sayHi(String msg) {
    System.out.println("\tsayHi:" + msg);
  }
 
  public void anotherSayHi(String msg) {
    this.sayHi(msg);
  }
 
}

最後來跑一把:

package com.cnblogs.yjmyzz.springbootdemo;
 
import com.cnblogs.yjmyzz.springbootdemo.service.HelloService;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
 
/**
 * @author 菩提樹下的楊過
 */
@ComponentScan("com.cnblogs.yjmyzz")
@Configuration
@EnableAspectJAutoProxy
public class SampleApplication {
 
  public static void main(String[] args) {
    AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SampleApplication.class);
    HelloService helloService = context.getBean(HelloService.class);
    helloService.sayHi("hi-1");
    System.out.println("\n");
    helloService.anotherSayHi("hi-2");
  }
}

輸出如下:

詳解spring中aop不生效的幾種解決辦法

顯然HelloService中的anotherSayHi方法,並未被aop增強。 原因其實很簡單,瞭解AOP原理的同學想必都知道,AOP的實現有二類,如果是基於介面的,會採用動態代理,生成一個代理類,如果是基於類的,會採用CGLib生成子類,然後在子類中擴充套件父類中的方法。

詳解spring中aop不生效的幾種解決辦法

本文中HelloService並不是一個介面,所以從上圖的斷點中可以看出,當Spring執行時,HelloService被增加為...EnhancerBySpringCGLib...。但是當呼叫到anotherSayHi時

詳解spring中aop不生效的幾種解決辦法

方法的呼叫方,其實是原始的HelloSerfvice例項,即:是未經過Spring AOP增強的物件例項。所以解決問題的思路就有了,想辦法用增強後的HelloService例項來呼叫!

方法一:用Autowired 注入自身的例項

詳解spring中aop不生效的幾種解決辦法

這個方法,第一眼看上去感覺有些怪,自己注入自己,感覺有點象遞迴/死迴圈的搞法,但確實可以work,Spring在解決迴圈依賴上有自己的處理方式,避免了死迴圈。

方法二:從Spring上下文獲取增強後的例項引用

詳解spring中aop不生效的幾種解決辦法

原理與方法一其實類似,不多解釋。

方法三: 利用AopContext

詳解spring中aop不生效的幾種解決辦法

不過這個方法要注意的是,主類入口上,必須加上exporseProxy=true,參考下圖:

詳解spring中aop不生效的幾種解決辦法

最後來驗證下這3種方法是否生效:

詳解spring中aop不生效的幾種解決辦法

從執行結果上看,3種方法都可以解決這個問題。 

到此這篇關於詳解spring中aop不生效的幾種解決辦法的文章就介紹到這了,更多相關spring中aop不生效內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!

作者:菩提樹下的楊過
出處:http://yjmyzz.cnblogs.com