一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Spring Boot之AOP配自定義注解的最佳實踐過程

Spring Boot之AOP配自定義注解的最佳實踐過程

2021-06-11 13:40JackieZheng Java教程

這篇文章主要給大家介紹了關于Spring Boot之AOP配自定義注解的最佳實踐過程,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

前言

aop(aspect oriented programming),即面向切面編程,是spring框架的大殺器之一。

首先,我聲明下,我不是來系統介紹什么是aop,更不是照本宣科講解什么是連接點、切面、通知和切入點這些讓人頭皮發麻的概念。

今天就來說說aop的一些應用場景以及如何通過和其他特性的結合提升自己的靈活性。下面話不多說了,來一起看看詳細的介紹吧

aop應用舉例

aop的一大好處就是解耦。通過切面,我們可以將那些反復出現的代碼抽取出來,放在一個地方統一處理。

同時,抽出來的代碼很多是與業務無關的,這樣可以方便開發者更加專注自己的業務邏輯的開發。

一個aop的典型應用場景就是日志打印。

下面是一個極端情況的controller

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
@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("============打印日志結束============");
  return "hello jackie";
 }
 
 @getmapping(value = "/test1")
 public string test1(httpservletrequest request, string var1) {
  log.info("============打印日志開始============");
  log.info("url: " + request.getrequesturl().tostring());
  log.info("============打印日志結束============");
  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("============打印日志結束============");
//  int i = 1/0;
  if (1<2)
   throw new illegalargumentexception("exception");
  return "test2";
 }
}

hellocontroller中提供了三個http接口,由于業務需要,所以每次進入某個方法的時候都需要打印請求的相關信息。

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

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

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

如何使用aop

在spring boot項目中,只需要如下幾步,就可以輕松上手aop。

添加maven依賴

?
1
2
3
4
<dependency>
 <groupid>org.springframework.boot</groupid>
 <artifactid>spring-boot-starter-aop</artifactid>
</dependency>

編寫切面類

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@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("============打印日志結束============");
//  log.info("before....");
 }
}

結果驗證對比

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

未使用切面功能打印日志

Spring Boot之AOP配自定義注解的最佳實踐過程

使用切面功能打印日志

Spring Boot之AOP配自定義注解的最佳實踐過程

從上面的結果展示發現,最終的效果是一樣的,但是使用切面更加簡潔,而且可復用。

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

這是為什么呢?

aop的局限

在切面類demoaspect中,我們看到了切入點的設置

?
1
2
@pointcut("execution(public * com.jackie.springbootdemo.controller.hellocontroller.test*(..))")
public void addadvice(){}

其中pointcut后面的表達式是用于控制切面的有效影響范圍。

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

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

好了,明白了表達式的含義,我們自然就看到了aop的局限性。

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

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

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

自定義注解配合aop

新建一個自定義注解

?
1
2
3
4
5
6
@documented
@retention(retentionpolicy.runtime)
@target(elementtype.method)
public @interface demoannotation {
 
}

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

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

?
1
2
3
4
5
6
7
8
9
10
11
@demoannotation
 @getmapping(value = "/test2")
 public string test2(httpservletrequest request, string var1, string var2) {
//  log.info("============打印日志開始============");
//  log.info("url: " + request.getrequesturl().tostring());
//  log.info("============打印日志結束============");
//  int i = 1/0;
  if (1<2)
   throw new illegalargumentexception("exception");
  return "test2";
 }

在切面類中將切入點的表達式改為

?
1
2
@pointcut("execution(public * com.jackie.springbootdemo.controller.*.*(..)) && @annotation(com.jackie.springbootdemo.annotation.demoannotation)")
public void addadvice(){}

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

Spring Boot之AOP配自定義注解的最佳實踐過程

從結果可以看出,訪問http://localhost:8080/test1?var1=1并沒有經過切面處理,因為不滿足切入點中的表達式要求。

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

代碼已提交至rome

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。

原文鏈接:http://www.cnblogs.com/bigdataZJ/p/springboot-aop.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 乳环贵妇堕落开发调教番号 | 亚洲国产视频一区 | 国产精品片 | 日本在线视频免费观看 | 免费xxxx日本大片在线观看 | www.青草视频 | 色戒 完整版 | 日韩操比视频 | gav男人天堂 | 国产成人亚洲综合网站不卡 | 暖暖视频免费观看视频中国.韩剧 | 黑人巨大初黑人解禁作品 | 欧美交换乱理伦片120秒 | 爽爽窝窝午夜精品一区二区 | 操bb| 国产一区二区精品 | 性印度freehd| 久久艹影院 | 武侠艳妇屈辱的张开双腿 | 91九色porny国产美女一区 | 精品一区二区三区 不卡高清 | 日本sss在线高清观看 | 67194在线免费观看 | 婷婷综合七月激情啪啪 | 国产成人在线视频 | 青青青国产成人久久111网站 | 五月天国产视频 | 欧美国产日产精品免费视频 | 嫩草视频在线观看视频播放 | 王王的视频ivk | 国产精品一区二区久久不卡 | 西西人体大胆啪啪私拍色约约 | 丝瓜草莓香蕉绿巨人幸福宝 | 99精品国产成人一区二区 | 青草碰人人澡人人澡 | 天天操网 | 久久久久免费视频 | 熟睡中的麻麻大白屁股小说 | 香蕉在线精品亚洲第一区 | 国产成人福利色视频 | 国产18在线 |