溫馨提示×

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

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

Spring AOP如何實(shí)現(xiàn)“切面式”valid校驗(yàn)

發(fā)布時(shí)間:2021-05-21 11:40:19 來(lái)源:億速云 閱讀:130 作者:小新 欄目:編程語(yǔ)言

這篇文章主要介紹Spring AOP如何實(shí)現(xiàn)“切面式”valid校驗(yàn),文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

why:

為什么要用aop實(shí)現(xiàn)校驗(yàn)?

answer:

spring mvc 默認(rèn)自帶的校驗(yàn)機(jī)制 @Valid + BindingResult, 但這種默認(rèn)實(shí)現(xiàn)都得在Controller方法的中去接收BindingResult,從而進(jìn)行校驗(yàn).

eg:

if (result.hasErrors()) {
 List<ObjectError> allErrors = result.getAllErrors();
 List<String> errorlists = new ArrayList<>();
  for (ObjectError objectError : allErrors) {
    errorlists.add(objectError.getDefaultMessage());
  }
 }

獲取errorlists。這樣實(shí)現(xiàn)的話,每個(gè)需要校驗(yàn)的方法都得重復(fù)調(diào)用,即使封裝也是。

可能上面那么說(shuō)還不能表明spring 的@Valid + BindingResult實(shí)現(xiàn),我先舉個(gè)“栗子”。

1. 栗子(舊版本)

1.1 接口層(IDAL)

eg: 簡(jiǎn)單的POST請(qǐng)求,@RequestBody接收請(qǐng)求數(shù)據(jù),@Valid + BindingResult進(jìn)行校驗(yàn)

  1. httpMethid: POST

  2. parameters:@RequestBody接收請(qǐng)求數(shù)據(jù)

  3. valid:@Valid +BindingResult

@ResponseBody
 @PostMapping("body")
 public ResponseVO bodyPost(@RequestBody @Valid TestVO body,BindingResult result){
  //校驗(yàn)到錯(cuò)誤
  if (result.hasErrors()) {
   List<ObjectError> allErrors = result.getAllErrors();
   List<String> lists = new ArrayList<>();
   for (ObjectError objectError : allErrors) {
     lists.add(objectError.getDefaultMessage());
   }
   return new ResponseVO(HttpStatus.BAD_REQUEST.value(), "parameter empty", lists);
 }
   return new ResponseVO(HttpStatus.OK.value(), "bodyPost", null);
}

1.2 實(shí)體(vo)校驗(yàn)內(nèi)容

@Valid + BindingResult的校驗(yàn)注解一大堆,網(wǎng)上一摸就有的!

public class TestVO {
  @Getter
  @Setter
  @Min(value = 0,message = "請(qǐng)求參數(shù)isString不能小于0")
  private Integer isInt;
  @Getter
  @Setter
  @NotBlank(message = "請(qǐng)求參數(shù)isString不能為空")
  private String isString;
}

1.3 結(jié)果測(cè)試

Spring AOP如何實(shí)現(xiàn)“切面式”valid校驗(yàn)

2. aop校驗(yàn)(升級(jí)版)

可以看到若是多個(gè)像bodyPost一樣都需要對(duì)body進(jìn)行校驗(yàn)的話,那么有一坨代碼就必須不斷復(fù)現(xiàn),即使改為父類(lèi)可復(fù)用方法,也得去調(diào)用。所以左思右想還是覺(jué)得不優(yōu)雅。所以有了aop進(jìn)行切面校驗(yàn)。

2.1 接口層(IDAL)

是的!你沒(méi)看錯(cuò),上面那一坨代碼沒(méi)了,也不需要調(diào)用父類(lèi)的的共用方法。就單單一個(gè)注解就完事了:@ParamValid

@ParamValid
@ResponseBody
@PostMapping("body")
public ResponseVO bodyPost(@RequestBody @Valid TestVO body,BindingResult result){
  return new ResponseVO("bodyPost", null);
}

2.2 自定義注解(annotation)

這個(gè)注解也是簡(jiǎn)簡(jiǎn)單單的用于方法的注解。

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ParamValid {}

2.3 重點(diǎn)!切面實(shí)現(xiàn)(Aspect)

切面詳解:

@Before: 使用注解方式@annotation(XX),凡是使用到所需切的注解(@ParamValid),都會(huì)調(diào)用該方法

JoinPoint: 通過(guò)JoinPoint獲取方法的參數(shù),以此獲取BindingResult所校驗(yàn)到的內(nèi)容

遷移校驗(yàn)封裝: 將原先那一坨校驗(yàn)遷移到Aspect中:validRequestParams

響應(yīng)校驗(yàn)結(jié)果:

  1. 通過(guò)RequestContextHolder獲取response

  2. 獲取響應(yīng)OutputStream

  3. 將BindingResult封裝響應(yīng)

@Aspect
@Component
public class ParamValidAspect {

  private static final Logger log = LoggerFactory.getLogger(ParamValidAspect.class);

  @Before("@annotation(paramValid)")
  public void paramValid(JoinPoint point, ParamValid paramValid) {
    Object[] paramObj = point.getArgs();
    if (paramObj.length > 0) {
      if (paramObj[1] instanceof BindingResult) {
        BindingResult result = (BindingResult) paramObj[1];
        ResponseVO errorMap = this.validRequestParams(result);
        if (errorMap != null) {
          ServletRequestAttributes res = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
          HttpServletResponse response = res.getResponse();
          response.setCharacterEncoding("UTF-8");
          response.setContentType(MediaType.APPLICATION_JSON_UTF8_VALUE);
          response.setStatus(HttpStatus.BAD_REQUEST.value());

          OutputStream output = null;
          try {
            output = response.getOutputStream();
            errorMap.setCode(null);
            String error = new Gson().toJson(errorMap);
            log.info("aop 檢測(cè)到參數(shù)不規(guī)范" + error);
            output.write(error.getBytes("UTF-8"));
          } catch (IOException e) {
            log.error(e.getMessage());
          } finally {
            try {
              if (output != null) {
                output.close();
              }
            } catch (IOException e) {
              log.error(e.getMessage());
            }
          }
        }
      }
    }
  }

  /**
   * 校驗(yàn)
   */
  private ResponseVO validRequestParams(BindingResult result) {
    if (result.hasErrors()) {
      List<ObjectError> allErrors = result.getAllErrors();
      List<String> lists = new ArrayList<>();
      for (ObjectError objectError : allErrors) {
        lists.add(objectError.getDefaultMessage());
      }
      return new ResponseVO(HttpStatus.BAD_REQUEST.value(), "parameter empty", lists);
    }
    return null;
  }
}

2.4 測(cè)試結(jié)果

 Spring AOP如何實(shí)現(xiàn)“切面式”valid校驗(yàn)

以上是“Spring AOP如何實(shí)現(xiàn)“切面式”valid校驗(yàn)”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

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

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

AI