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

spring boot 使用Aop通知打印控制器請求報文和返回報文問題

 更新時間:2018年04月11日 08:44:58   作者:淺淺_淺默  
這篇文章主要介紹了spring-boot 使用Aop通知打印控制器請求報文和返回報文,非常不錯,具有參考借鑒價值,需要的朋友可以參考下

一、簡介

開發(fā)過程中我們往往需要寫許多例如: 

@GetMapping("/id/get")
  public Result getById( String id) throws Exception{
    log.info("請求參數(shù)為:"+id);
    verify(new VerifyParam("部門id", id));
    Result result = new Result("通過id獲取部門信息成功!", service.queryById(id));
    log.info("返回報文為:"+result.toString());
    return result;
  }

打印請求參數(shù)以及返回參數(shù)的方法,而這些操作存在于每個方法之中,使得我們代碼較為冗余,為此我們可以通過動態(tài)代理將打印參數(shù)和打印返回報文作為切面,使用切入點(diǎn)表達(dá)式將其切入至每個方法之中。

二、步驟

1、引入Aop相關(guān)的依賴:

<!--AOP相關(guān)的依賴-->
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

引入依賴后spring-aop會加載其需要的依賴,spring默認(rèn)使用aspectJ實現(xiàn)通知

其中aspectjweaver.jar中包含了解析aspectJ切入點(diǎn)表達(dá)式的文件,使用切入點(diǎn)表達(dá)式處理事務(wù)的時候也需要加入此依賴。

2、配置:

1)、創(chuàng)建配置類:

/**
 * @功能描述:用于controller層操作的AOP類
 * @author Administrator
 */
@Component // 將對象交由spring進(jìn)行管理
@Aspect // 代表此類為一個切面類
public class ControllerAop {
}

其中@Aspect 注解代表其為一個切面管理類,可以在其下定義切入點(diǎn)表達(dá)式,aspectJ框架會進(jìn)行解析。

2)、定義切入點(diǎn)表達(dá)式:

@Pointcut("execution(public * com.hzt.manage.*.web.controller..*.*(..))") // 切入點(diǎn)表達(dá)式
  public void privilege() {
  }

其中,@Pointcut代表此方法為一個切入點(diǎn)表達(dá)式。其value值為切入點(diǎn)表達(dá)式,其中value可以省略其大致格式為:

  @注解(表達(dá)標(biāo)簽+表達(dá)式格式)
的格式,Spring AOP支持的AspectJ切入點(diǎn)指示符如下:

1、 execution:用于匹配方法執(zhí)行的連接點(diǎn);

2、within:用于匹配指定類型內(nèi)的方法執(zhí)行;

3、this:用于匹配當(dāng)前AOP代理對象類型的執(zhí)行方法;注意是AOP代理對象的類型匹配,這樣就可能包括引入接口也類型匹配;

4、target:用于匹配當(dāng)前目標(biāo)對象類型的執(zhí)行方法;注意是目標(biāo)對象的類型匹配,這樣就不包括引入接口也類型匹配;

5、args:用于匹配當(dāng)前執(zhí)行的方法傳入的參數(shù)為指定類型的執(zhí)行方法;

6、@within:用于匹配所以持有指定注解類型內(nèi)的方法;

7、@target:用于匹配當(dāng)前目標(biāo)對象類型的執(zhí)行方法,其中目標(biāo)對象持有指定的注解;

8、@args:用于匹配當(dāng)前執(zhí)行的方法傳入的參數(shù)持有指定注解的執(zhí)行;

9、@annotation:用于匹配當(dāng)前執(zhí)行方法持有指定注解的方法;

10、bean:Spring AOP擴(kuò)展的,AspectJ沒有對于指示符,用于匹配特定名稱的Bean對象的執(zhí)行方法;

11、reference pointcut:表示引用其他命名切入點(diǎn),只有@ApectJ風(fēng)格支持,Schema風(fēng)格不支持。

args中定義了切入點(diǎn)表達(dá)式方法執(zhí)行時候的參數(shù):

  @Pointcut(value="execution(public * com.hzt.manage.*.web.controller..*.*(..))&&args(param)",argNames="param") // 切入點(diǎn)表達(dá)式
  public void privilege1(String param) {
  }

我們重點(diǎn)介紹 execution 方法連接點(diǎn)的表達(dá)式,其大概結(jié)構(gòu)為:

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)throws-pattern?)

1、修飾符匹配(modifier-pattern?)(可省略)

2、返回值匹配(ret-type-pattern)可以為*表示任何返回值 ,如 (String) 代表只篩選返回String類型的切入點(diǎn) ,全路徑的類名等(不可省略)

3、類路徑匹配(declaring-type-pattern?)如*.manage代表一級包為任意,二級包為manage的名稱。*..manage代表所有manage包下的子類包。com..*.comtroller代表com包下所有的controller包等,*代表所有包都匹配。(不可省略)

4、方法名匹配(name-pattern)可以指定方法名 或者 *代表所有, get* 代表以get開頭的所有方法,也可指定前綴*get代表任意后綴為get的方法(不可省略)

5、參數(shù)匹配((param-pattern))可以指定具體的參數(shù)類型,多個參數(shù)間用“,”隔開,各個參數(shù)也可以用“*”來表示匹配任意類型的參數(shù),如(String)表示匹配一個String參數(shù)的方法;(*,String) 表示匹配有兩個參數(shù)的方法,第一個參數(shù)可以是任意類型,而第二個參數(shù)是String類型;可以用(..)表示任意參數(shù)(不可省略)

6、異常類型匹配(throws-pattern?)

3、定義切面方法

@Around("privilege()")
  public Object around(ProceedingJoinPoint pjd) throws Throwable {
    // 獲取方法名
    String className = pjd.getSignature().getClass().getName();
    // 獲取執(zhí)行的方法名稱
    String methodName = pjd.getSignature().getName();
    /** 初始化日志打印 */
    Logger log = LoggerFactory.getLogger(className);
    // 定義返回參數(shù)
    Object result = null;
    // 記錄開始時間
    long start = System.currentTimeMillis();
    // 獲取方法參數(shù)
    Object[] args = pjd.getArgs();
    String params = "前端請求參數(shù)為:";
    //獲取請求參數(shù)集合并進(jìn)行遍歷拼接
    for (Object object : args) {
      params += object.toString() + ",";
    }
    params = params.substring(0, params.length() - 1);
    //打印請求參數(shù)參數(shù)
    log.info(className+"類的"+methodName + "的" + params);
    // 執(zhí)行目標(biāo)方法
    result = pjd.proceed();
    // 打印返回報文
    log.info("方法返回報文為:" + (result instanceof Result ? (Result) result : result));
    // 獲取執(zhí)行完的時間
    log.info(methodName + "方法執(zhí)行時長為:" + (System.currentTimeMillis() - start));
    return result;
  }

5、@Around 環(huán)繞通知,如上代碼所示便是環(huán)繞通知,其有ProceedingJoinPoint參數(shù)

其中 pjd.proceed();方法代表去執(zhí)行目標(biāo)方法,并獲得一個Object類型的返回值 ,我們可以對返回值進(jìn)行加工處理,如裝飾加工等。

return的值為方法執(zhí)行的結(jié)果。上述代碼中首先獲取類名、方法名、方法請求參數(shù)等,進(jìn)行打印的拼接,并且記錄方法執(zhí)行的開始時間,并進(jìn)行打印至日志。

然后執(zhí)行方法,獲取到方法返回結(jié)果,進(jìn)行打印執(zhí)行時間和執(zhí)行結(jié)果。

最后返回執(zhí)行結(jié)果。即使用Aop打印請求報文和返回報文的aop切面編碼結(jié)束。

其中@Around代表其為一個環(huán)繞通知方法,其有以下幾種類型:

1、@Before前置通知,擁有請求參數(shù) JoinPoint ,用來連接當(dāng)前連接點(diǎn)的連接細(xì)節(jié),一般包括方法名和參數(shù)值。在方法執(zhí)行前進(jìn)行執(zhí)行方法體,不能改變方法參數(shù),也不能改變方法執(zhí)行結(jié)果。

  @Before(value = "privilege()")
  public void before(JoinPoint joinPoint) {
  }

2、@After 后置通知:在目標(biāo)方法執(zhí)行之后,無論是否發(fā)生異常,都進(jìn)行執(zhí)行的通知。在后置通知中,不能訪問目標(biāo)方法的執(zhí)行結(jié)果(因為有可能發(fā)生異常),不能改變方法執(zhí)行結(jié)果。

@Before(value = "privilege()")
  public void after(JoinPoint joinPoint) {    
  }

3、@AfterReturning 返回通知,在目標(biāo)方法執(zhí)行結(jié)束時,才執(zhí)行的通知,同后置方法相同。其能訪問方法執(zhí)行結(jié)果(因為正常執(zhí)行)和方法的連接細(xì)節(jié),但是不能改變方法執(zhí)行結(jié)果。

  @AfterReturning(value = "privilege()")
  public void afterReturing(JoinPoint joinPoint,Object result) {
  }

result中存放的為方法的返回值。

4、@AfterThrowing 異常通知:在目標(biāo)方法出現(xiàn)異常時才會進(jìn)行執(zhí)行的代碼。 throwing屬性代表方法體執(zhí)行時候拋出的異常,其值一定與方法中Exception的值需要一致。

  @AfterThrowing(value="privilege()",throwing="ex")
  public void exce(JoinPoint joinPoint, Exception ex) {    
  }

三、測試

編寫一個Controller方法

@RestController
@RequestMapping("/api/v1/dept")
public class DeptController extends BaseController{
  /** 日志記錄類 */
  private Logger log = LoggerFactory.getLogger(getClass());
  /** 自家的service */
  @Autowired
  private DeptService service;
  /**
   * @功能描述:根據(jù)id查詢部門內(nèi)容的方法
   * @return Dept
   */
  @GetMapping("/id/get")
  public Result getById( String id) throws Exception{
    verify(new VerifyParam("部門id", id));
    return new Result("通過id獲取部門信息成功!", service.queryById(id));
  }
}

如此我們的controller層中的方法就大大的簡潔了。

測試結(jié)果:

2018-04-10 22:59:27.468  INFO 1460 --- [nio-8088-exec-5] nProceedingJoinPoint$MethodSignatureImpl : getById的前端請求參數(shù)為:22
2018-04-10 22:59:27.470  INFO 1460 --- [nio-8088-exec-5] nProceedingJoinPoint$MethodSignatureImpl : 方法返回報文為:Result [result_code=suc, result_message=通過id獲取部門信息成功!, data=Dept [id=22, no=22, name=22, manager=22, description=22, phone=22, createTime=Thu Apr 19 23:38:37 CST 2018, editTime=null]]
2018-04-10 22:59:27.470  INFO 1460 --- [nio-8088-exec-5] nProceedingJoinPoint$MethodSignatureImpl : getById方法執(zhí)行時長為:2

如此便能很雅觀簡潔隱式的打印請求參數(shù)、返回結(jié)果和執(zhí)行時間等!

相關(guān)文章

  • java this 用法詳解及簡單實例

    java this 用法詳解及簡單實例

    這篇文章主要介紹了java this 用法詳解及簡單實例的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • SpringBoot使用自定義json解析器的使用方法

    SpringBoot使用自定義json解析器的使用方法

    本篇文章主要介紹了SpringBoot使用自定義json解析器的使用方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-04-04
  • SpringCloud Gateway使用詳解

    SpringCloud Gateway使用詳解

    Spring Cloud Gateway是一個基于Spring Boot 2.x和Spring WebFlux的API網(wǎng)關(guān),可以幫助我們構(gòu)建微服務(wù)架構(gòu)中的統(tǒng)一入口。感興趣的同學(xué)可以參考一下
    2023-04-04
  • Java使用openOffice對于word的轉(zhuǎn)換及遇到的問題解決

    Java使用openOffice對于word的轉(zhuǎn)換及遇到的問題解決

    開發(fā)過程中經(jīng)常會使用java將office系列文檔轉(zhuǎn)換為PDF, 一般都使用微軟提供的openoffice+jodconverter 實現(xiàn)轉(zhuǎn)換文檔,下面這篇文章主要給大家介紹了關(guān)于Java通過openOffice對于word的轉(zhuǎn)換及遇到問題的解決方法,需要的朋友可以參考下
    2018-09-09
  • java?io文件操作刪除文件或文件夾的7種方法

    java?io文件操作刪除文件或文件夾的7種方法

    這篇文章主要為大家介紹了java?io文件操作刪除文件或文件夾的7種方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步
    2022-03-03
  • Java簡單統(tǒng)計字符串中漢字,英文字母及數(shù)字?jǐn)?shù)量的方法

    Java簡單統(tǒng)計字符串中漢字,英文字母及數(shù)字?jǐn)?shù)量的方法

    這篇文章主要介紹了Java簡單統(tǒng)計字符串中漢字,英文字母及數(shù)字?jǐn)?shù)量的方法,涉及java針對字符串的遍歷、編碼轉(zhuǎn)換、判斷等相關(guān)操作技巧,需要的朋友可以參考下
    2017-06-06
  • Java實現(xiàn)24點(diǎn)小游戲

    Java實現(xiàn)24點(diǎn)小游戲

    這篇文章主要為大家詳細(xì)介紹了Java實現(xiàn)24點(diǎn)小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-01-01
  • java生成json數(shù)據(jù)示例

    java生成json數(shù)據(jù)示例

    這篇文章主要介紹了java生成json數(shù)據(jù)示例,需要的朋友可以參考下
    2014-04-04
  • mybatis映射文件操作存儲過程的實現(xiàn)

    mybatis映射文件操作存儲過程的實現(xiàn)

    本文主要介紹了mybatis映射文件操作存儲過程的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Java中圖片的常用操作代碼總結(jié)

    Java中圖片的常用操作代碼總結(jié)

    這篇文章主要為大家詳細(xì)介紹了Java中對圖片進(jìn)行常用操作處理的代碼,例如生成自定義圖片、獲取圖片格式、圖片的裁剪與壓縮等,感興趣的小伙伴可以了解一下
    2022-11-11

最新評論