SpringBoot项目使用 validation进行数据校验

validation进行数据校验

@Validated 注解和 @Valid 注解都是 Spring Framework 中用于数据校验的注解,但它们有以下几点区别:

  1. 所在包路径不同:@Valid 注解位于 javax.validation.constraints 包下,而 @Validated 注解位于 org.springframework.validation.annotation 包下。

  2. 支持的目标类型不同:@Valid 注解支持作用在方法参数、属性、方法返回值等多种目标类型上,而 @Validated 注解仅支持作用在方法参数和类上。

  3. 校验能力不同:@Valid 注解使用 Bean Validation 规范进行数据校验,支持约束注解如 @NotNull@Email@Pattern 等;@Validated 注解则使用 Spring Validation 框架进行数据校验,支持约束注解如 @NotBlank@Length@Min 等。虽然两者都可以用于数据校验,但这两种框架的实现方式略有不同。

  4. 使用场景不同:@Valid 注解通常适用于在 Java EE 容器中使用,例如在 JPA 实体或 EJB 中进行数据校验;而 @Validated 注解则主要适用于 Spring Web 应用程序中,在 Controller 或 Service 层中对请求参数进行数据校验。

需要注意的是,在使用 @Valid 注解进行数据校验时,还需要在应用程序中添加 Bean Validation API 和实现,以保证数据校验能够正常工作。而在使用 @Validated 注解进行数据校验时,Spring 框架会自动将约束注解转换为 Spring 内部的校验器,并使用其中的校验逻辑进行数据校验。

当您一个controller类中存在多个需要校验的,但是不想自己一个一个加就可以直接在controller类上加上@Validated注解

校验注解与描述

校验注解 描述
@Null 验证对象是否为null
@NotNull 验证对象是否不为null, 与@Null相反
@AssertTrue 验证Boolean类型字段是否为true
@AssertFalse 验证Boolean类型字段是否为false
@Min(value) 验证数字是否大于等于指定的最小值
@Max(value) 验证数字是否小于等于指定的最大值
@DecimalMin(value) 验证字符串表示的数字是否大于等于指定的最小值
@DecimalMax(value) 验证字符串表示的数字是否小于等于指定的最大值
@Size(max=, min=) 验证字符串、集合、Map、数组的大小是否在指定范围内
@Digits(integer, fraction) 验证数值是否符合指定的格式,integer指定整数精度,fraction指定小数精度
@Past 验证日期是否在当前时间之前
@Future 验证日期是否在当前时间之后
@Pattern(regex=, flag=) 验证字符串是否符合指定的正则表达式
@NotEmpty 验证字符串是否非空
@NotBlank 验证字符串是否非空白字符
@Email 验证字符串是否为邮箱地址格式

maven导入

spring-boot-starter-validation 依赖项是 Spring Boot 提供的用于支持 Bean Validation API 的快速启动器。它包含了 Bean Validation API 和 Hibernate Validator 实现等必要的依赖项,并且已经在 Spring Boot 中预先配置好了相关的参数和属性。

<dependencies>
	<dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-validation</artifactId>
	</dependency>
</dependencies>

具体使用代码:

UserRegisterDto 类,用于接收前端传入的内容:
注意:里面用到了groups,这里是为了方便在接收登录或者注册的时候用Validated(group)来指定哪几个校验注解生效

@Data
public class UserVerifyDto {
    @Email(groups = {RegisterGroup.class, LoginGroup.class, Email.class},regexp = "\\w+([-+.]\\w+)*@\\w+([-.]\\w+)*\\.\\w+([-.]\\w+)*", message = "邮箱格式错误")
    private String email;
    @Length(groups = {RegisterGroup.class, LoginGroup.class}, min = 2, max = 10, message = "昵称建议在2-10之间的长度")
    private String nickname;
    @Length(groups = {RegisterGroup.class, LoginGroup.class}, min =6, message = "密码最少要6位数")
    private String password;
    @Length(groups = {RegisterGroup.class}, min = 5, max = 5, message = "请输入正确的验证码")
    private String code;
}

下面的方法里的形参有带入@Validated注解,该注解可让spring框架知道要进行校验。若使用的是Valid,则无法使用分组

@PostMapping("register")
public R register(@RequestBody @Validated(RegisterGroup.class) UserVerifyDto registerDto) {
    CacheObject<String> verifyCodeCache = REGISTER_EMAIL_CACHE.get(registerDto.getEmail());
    if (verifyCodeCache == null ||
            verifyCodeCache.isExpire() ||
            !registerDto.getCode().equals(verifyCodeCache.getObj())) {
        return R.error().msg("请获取验证码,该验证码已失效!");
    }
    return userService.registerUser(registerDto);
}

上面定义好校验注解后,我们还需要处理它返回的校验错误结果:
1.通过捕获异常处理
2.通过在方法形参里加入BindingResult

1通过捕获异常处理校验结果

@RestControllerAdvice
public class GlobalExceptionHandler {
    @ExceptionHandler(BindException.class)
    public R handleValidationException(BindException ex) {
        R result = R.error();
        for (FieldError error : ex.getBindingResult().getFieldErrors()) {
            result.data(error.getField(), error.getDefaultMessage());
        }
        Optional<Object> firstError = result.getData().values().stream().findFirst();
        return result.msg("请完善信息: "+firstError.orElse("表单内容") +
                (ex.getErrorCount()>1?" 等多项问题":""));
    }
}

2.通过在方法形参里加入BindingResult

@RestController
@RequestMapping("/users")
@Validated
public class UserController {
    @PostMapping
    public String createUser(@RequestBody User user, BindingResult result) {
        if (result.hasErrors()) {
            // 处理错误信息
            return "error";
        }
        // 处理用户信息
        return "success";
    }
}
posted @ 2023-05-01 13:30  LiusCraft  阅读(1134)  评论(0编辑  收藏  举报