溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Spring Boot之AOP配自定義注解的示例分析

發(fā)布時間:2021-08-11 10:58:10 來源:億速云 閱讀:114 作者:小新 欄目:編程語言

這篇文章主要介紹了Spring Boot之AOP配自定義注解的示例分析,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

前言

AOP(Aspect Oriented Programming),即面向切面編程,是Spring框架的大殺器之一。

首先,我聲明下,我不是來系統(tǒng)介紹什么是AOP,更不是照本宣科講解什么是連接點(diǎn)、切面、通知和切入點(diǎn)這些讓人頭皮發(fā)麻的概念。

AOP應(yīng)用舉例

AOP的一大好處就是解耦。通過切面,我們可以將那些反復(fù)出現(xiàn)的代碼抽取出來,放在一個地方統(tǒng)一處理。

同時,抽出來的代碼很多是與業(yè)務(wù)無關(guān)的,這樣可以方便開發(fā)者更加專注自己的業(yè)務(wù)邏輯的開發(fā)。

一個AOP的典型應(yīng)用場景就是日志打印。

下面是一個極端情況的Controller

@RestController
@RequestMapping("/")
public class HelloController {
 private static final Logger LOG = LoggerFactory.getLogger(HelloController.class);

 @GetMapping(value = "/index")
 public String index(HttpServletRequest request) {
  LOG.info("============打印日志開始============");
  LOG.info("URL: " + request.getRequestURL().toString());
  LOG.info("============打印日志結(jié)束============");
  return "hello jackie";
 }

 @GetMapping(value = "/test1")
 public String test1(HttpServletRequest request, String var1) {
  LOG.info("============打印日志開始============");
  LOG.info("URL: " + request.getRequestURL().toString());
  LOG.info("============打印日志結(jié)束============");
  return "test1";
 }

 @DemoAnnotation
 @GetMapping(value = "/test2")
 public String test2(HttpServletRequest request, String var1, String var2) {
  LOG.info("============打印日志開始============");
  LOG.info("URL: " + request.getRequestURL().toString());
  LOG.info("============打印日志結(jié)束============");
//  int i = 1/0;
  if (1<2)
   throw new IllegalArgumentException("exception");
  return "test2";
 }
}

HelloController中提供了三個Http接口,由于業(yè)務(wù)需要,所以每次進(jìn)入某個方法的時候都需要打印請求的相關(guān)信息。

當(dāng)然,如果只是上面的例子,我們完全可以通過其他手段讓代碼看著并不這么糟糕。我們可以抽象一個打印方法,將相同的代碼封裝在這個方法中,之后在各個方法中每次調(diào)用即可。

但是,這種處理方法似乎抽象的還不夠,因為我們在每個Http接口中還是要調(diào)用這個抽象的函數(shù)。而且,比較要命的是,這打印日志的代碼與其他業(yè)務(wù)代碼顯得有些格格不入。

所以,這時候,我們想到了AOP。

如何使用AOP

在Spring Boot項目中,只需要如下幾步,就可以輕松上手AOP。

添加maven依賴

<dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

編寫切面類

@Aspect
@Component
public class DemoAspect {
 private static final Logger LOG = LoggerFactory.getLogger(DemoAspect.class);

 @Pointcut("execution(public * com.jackie.springbootdemo.controller.HelloController.test*(..))")
 public void addAdvice(){}

 @Before("addAdvice()")
 public void before(JoinPoint joinPoint){
  Object[] args = joinPoint.getArgs();
  HttpServletRequest requests = (HttpServletRequest) args[0];
  LOG.info("============打印日志開始============");
  LOG.info("URL: " + requests.getRequestURL().toString());
  LOG.info("============打印日志結(jié)束============");
//  LOG.info("before....");
 }
}

結(jié)果驗證對比

啟動SpringBootDemoApplication,訪問url:http://localhost:8080/test2?var1=1&var2=2

未使用切面功能打印日志

Spring Boot之AOP配自定義注解的示例分析

使用切面功能打印日志

Spring Boot之AOP配自定義注解的示例分析

從上面的結(jié)果展示發(fā)現(xiàn),最終的效果是一樣的,但是使用切面更加簡潔,而且可復(fù)用。

如上訪問的是test2接口,如果訪問test1接口也可以走切面類實現(xiàn)打印日志的需求,但是如果走index請求就不會打印日志了。

這是為什么呢?

AOP的局限

在切面類DemoAspect中,我們看到了切入點(diǎn)的設(shè)置

@Pointcut("execution(public * com.jackie.springbootdemo.controller.HelloController.test*(..))")
public void addAdvice(){}

其中Pointcut后面的表達(dá)式是用于控制切面的有效影響范圍。

**表達(dá)式中,第一個表示返回任意類型,第二個表示任意方法名,后面的小括號表示任意參數(shù)值,這里是以test為前綴的,所以可以匹配上test1和test2方法。

注意,在第二個之前也可以再有個,即HelloController所在位置,表示任意類名,假如這里是有兩個*.則表示包括包里面的子包。**

好了,明白了表達(dá)式的含義,我們自然就看到了AOP的局限性。

當(dāng)我們要使用切面前,就要寫好表達(dá)式,但是項目一直在做,代碼一直在加,那誰能保證后面接收代碼的兄弟也正好知道這個test前綴的意義這么重大呢?

如果他非要用hello作為前綴,那么本應(yīng)該匹配到的接口就匹配不上了,日志也就不能正常打印了。

這時候,自定義注解,就能夠很好的解決這個問題。

自定義注解配合AOP

新建一個自定義注解

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface DemoAnnotation {

}

自定義注解的花樣也很多,比如可以在注解中聲明變量等,但這些不是我們這次討論的重點(diǎn)。

將該注解添加到Http接口test2方法上

 @DemoAnnotation
 @GetMapping(value = "/test2")
 public String test2(HttpServletRequest request, String var1, String var2) {
//  LOG.info("============打印日志開始============");
//  LOG.info("URL: " + request.getRequestURL().toString());
//  LOG.info("============打印日志結(jié)束============");
//  int i = 1/0;
  if (1<2)
   throw new IllegalArgumentException("exception");
  return "test2";
 }

在切面類中將切入點(diǎn)的表達(dá)式改為

@Pointcut("execution(public * com.jackie.springbootdemo.controller.*.*(..)) && @annotation(com.jackie.springbootdemo.annotation.DemoAnnotation)")
public void addAdvice(){}

這樣,我們不需要限制在controller類中是以test作為前綴了,只要是在上面定義的類路徑下,并且掃描到注解DemoAnnotation就可以讓切面生效。

Spring Boot之AOP配自定義注解的示例分析

從結(jié)果可以看出,訪問http://localhost:8080/test1?var1=1并沒有經(jīng)過切面處理,因為不滿足切入點(diǎn)中的表達(dá)式要求。

這樣做的好處在于,控制的粒度更細(xì),也更加靈活,方便切面功能的實現(xiàn)和細(xì)分。

感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“Spring Boot之AOP配自定義注解的示例分析”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,更多相關(guān)知識等著你來學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI