介绍

JSR-380是 J2EE 的一个标准,用于校验实体属性,它是JSR-303的升级版,在 Spring Boot 中能够基于它优雅实现参数校验。

<!--more-->

示例

在没有应用JSR-380之前,咱们个别都会将参数校验硬编码在controller类中,示例:

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

而应用JSR-380只须要通过增加对应的注解即可实现校验,示例:

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

这样看起来代码是不是清新了很多,只须要在须要校验的字段上加上对应的校验注解,而后对须要校验的中央加上@Validated注解,而后框架就会帮咱们实现校验。

通过全局异样自定义谬误响应

框架校验失败之后会抛出异样,须要捕捉这个异样而后来自定义校验不通过的谬误响应,这里间接贴代码,兼容@RequestBody@ModelAttribute@RequestParam三种入参的校验:

@ControllerAdvicepublic class GlobalExceptionHandler {    @ExceptionHandler(value = {MethodArgumentNotValidException.class, BindException.class})    public ResponseEntity<Result> methodArgumentNotValidHandler(HttpServletRequest request, Exception e) {        BindingResult bindingResult;        if (e instanceof MethodArgumentNotValidException) {            //@RequestBody参数校验            bindingResult = ((MethodArgumentNotValidException) e).getBindingResult();        } else {            //@ModelAttribute参数校验            bindingResult = ((BindException) e).getBindingResult();        }        FieldError fieldError = bindingResult.getFieldError();        return ResponseEntity.ok(Result.fail(Result.CODE_PARAMS_INVALID, "[" + fieldError.getField() + "]" + fieldError.getDefaultMessage()));    }    //@RequestParam参数校验    @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 {            // 这个不是JSR规范返回的异样,要自定义提醒文本            field = ((MissingServletRequestParameterException) e).getParameterName();            msg = "不能为空";        }        return ResponseEntity.ok(Result.fail(Result.CODE_PARAMS_INVALID, "[" + field + "]" + msg));    }}

而后再拜访一下接口,能够看到谬误提醒曾经按自定义的标准显示了:

能够看到都不须要写任何提醒文本就能够实现校验和提醒,上图的不能为空是框架内置的I18N国际化反对,每个注解都内置相应的提醒模板。

罕用校验注解

注解形容
@NotNull验证值不为 null
@AssertTrue验证值为 true
@Size验证值的长度介于 min 和 max 之间,可利用于 String、Collection、Map 和数组类型
@Min验证值不小于该值
@Max验证值不大于该值
@Email验证字符串是无效的电子邮件地址
@NotEmpty验证值不为 null 或空,可利用于 String、Collection、Map 和数组类型
@NotBlank验证字符串不为 null 并且不是空白字符
@Positive验证数字为负数
@PositiveOrZero验证数字为负数(包含 0)
@Negative验证数字为正数
@NegativeOrZero验证数字为正数(包含 0)
@Past验证日期值是过来
@PastOrPresent验证日期值是过来(包含当初)
@Future验证日期值是将来
@FutureOrPresent验证日期值是将来(包含当初)

附录

本文残缺代码放在github。

  • Java Bean Validation Basics
  • JSR-380 标准
我是MonkeyWie,欢送扫码????????关注!不定期在公众号中分享JAVAGolang前端dockerk8s等干货常识。