Spring Boot使用JSR-380進(jìn)行校驗(yàn)

介紹

JSR-380是 J2EE 的一個(gè)規(guī)范,用于校驗(yàn)實(shí)體屬性,它是JSR-303的升級版,在 Spring Boot 中可以基于它優(yōu)雅實(shí)現(xiàn)參數(shù)校驗(yàn)。

示例

在沒有使用JSR-380之前,我們一般都會將參數(shù)校驗(yàn)硬編碼在controller類中,示例:

public Result add(@RequestBody User user){
    if(StringUtils.isBlank(user.getName())){
        return Result.error("用戶名不能為空");
    }
    // ...
}

而使用JSR-380只需要通過添加對應(yīng)的注解即可實(shí)現(xiàn)校驗(yàn),示例:

@Data
public class User{
    @NotBlank
    private String name;
    private Integer age;
}
public Result register(@Validated @RequestBody User user){
    // ...
}

這樣看起來代碼是不是清爽了很多,只需要在需要校驗(yàn)的字段上加上對應(yīng)的校驗(yàn)注解,然后對需要校驗(yàn)的地方加上@Validated注解,然后框架就會幫我們完成校驗(yàn)。

通過全局異常自定義錯(cuò)誤響應(yīng)

框架校驗(yàn)失敗之后會拋出異常,需要捕獲這個(gè)異常然后來自定義校驗(yàn)不通過的錯(cuò)誤響應(yīng),這里直接貼代碼,兼容@RequestBody、@ModelAttribute、@RequestParam三種入?yún)⒌男r?yàn):

@ControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler(value = {MethodArgumentNotValidException.class, BindException.class})
    public ResponseEntity<Result> methodArgumentNotValidHandler(HttpServletRequest request, Exception e) {
        BindingResult bindingResult;
        if (e instanceof MethodArgumentNotValidException) {
            //@RequestBody參數(shù)校驗(yàn)
            bindingResult = ((MethodArgumentNotValidException) e).getBindingResult();
        } else {
            //@ModelAttribute參數(shù)校驗(yàn)
            bindingResult = ((BindException) e).getBindingResult();
        }
        FieldError fieldError = bindingResult.getFieldError();
        return ResponseEntity.ok(Result.fail(Result.CODE_PARAMS_INVALID, "[" + fieldError.getField() + "]" + fieldError.getDefaultMessage()));
    }

    //@RequestParam參數(shù)校驗(yàn)
    @ExceptionHandler(value = {ConstraintViolationException.class, MissingServletRequestParameterException.class})
    public ResponseEntity<Result> constraintViolationHandler(Exception e) {
        String field;
        String msg;
        if (e instanceof ConstraintViolationException) {
            ConstraintViolation<?> constraintViolation = ((ConstraintViolationException) e).getConstraintViolations().stream().findFirst().get();
            List<Path.Node> pathList = StreamSupport.stream(constraintViolation.getPropertyPath().spliterator(), false)
                    .collect(Collectors.toList());
            field = pathList.get(pathList.size() - 1).getName();
            msg = constraintViolation.getMessage();
        } else {
            // 這個(gè)不是JSR標(biāo)準(zhǔn)返回的異常,要自定義提示文本
            field = ((MissingServletRequestParameterException) e).getParameterName();
            msg = "不能為空";
        }
        return ResponseEntity.ok(Result.fail(Result.CODE_PARAMS_INVALID, "[" + field + "]" + msg));
    }
}

然后再訪問一下接口,可以看到錯(cuò)誤提示已經(jīng)按自定義的規(guī)范顯示了:

可以看到都不需要寫任何提示文本就可以完成校驗(yàn)和提示,上圖的不能為空是框架內(nèi)置的I18N國際化支持,每個(gè)注解都內(nèi)置相應(yīng)的提示模板。

常用校驗(yàn)注解

注解 描述
@NotNull 驗(yàn)證值不為 null
@AssertTrue 驗(yàn)證值為 true
@Size 驗(yàn)證值的長度介于 min 和 max 之間,可應(yīng)用于 String、Collection、Map 和數(shù)組類型
@Min 驗(yàn)證值不小于該值
@Max 驗(yàn)證值不大于該值
@Email 驗(yàn)證字符串是有效的電子郵件地址
@NotEmpty 驗(yàn)證值不為 null 或空,可應(yīng)用于 String、Collection、Map 和數(shù)組類型
@NotBlank 驗(yàn)證字符串不為 null 并且不是空白字符
@Positive 驗(yàn)證數(shù)字為正數(shù)
@PositiveOrZero 驗(yàn)證數(shù)字為正數(shù)(包括 0)
@Negative 驗(yàn)證數(shù)字為負(fù)數(shù)
@NegativeOrZero 驗(yàn)證數(shù)字為負(fù)數(shù)(包括 0)
@Past 驗(yàn)證日期值是過去
@PastOrPresent 驗(yàn)證日期值是過去(包括現(xiàn)在)
@Future 驗(yàn)證日期值是未來
@FutureOrPresent 驗(yàn)證日期值是未來(包括現(xiàn)在)

附錄

本文完整代碼放在github。

本文首發(fā)于我的博客:https://monkeywie.cn,歡迎收藏!不定期分享JAVA、Golang前端、docker、k8s等干貨知識。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容