从零搭建开发脚手架 Spring Boot 输入参数校验多种方式整理

369 阅读6分钟

文章目录

准备做个《从零搭建开发脚手架系列》,把遇到的问题和搭建的过程记录分享给大家。

背景

当涉及到用户输入时,就需要对输入的内容做校验,例如:姓名不能为空,年龄范围为0-150等等。我们使用Spring Boot内置的验证来实现此功能。

从Spring Boot 2.3开始,我们需要显式添加spring-boot-starter-validation依赖项:

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

常见场景

整理分享下,在项目开发中常见的几种方式。

1.bean验证

待验证实体:

@Data
@Builder
public class LakerUser {
    /**
     * 姓名
     */
    @NotBlank(message = "姓名不能为空")
    private String name;

    /**
     * 年龄
     */
    @Min(value = 0, message = "年龄应大于等于0")
    @Min(value = 150, message = "年龄应小于等于150")
    private int age;

    /**
     * 手机号
     */
    @NotBlank(message = "手机号不能为空")
    @Pattern(regexp = "^[1][3,4,5,6,7,8,9][0-9]{9}$", message = "手机号格式有误")
    private String mobile;

}

控制器:

方法入参:加上@Validated

    @RequestMapping("/lakeruser")
    public Response validatorBean(@RequestBody @Validated LakerUser lakerUser) {
        return Response.ok();
    }

全局异常处理:

@RestControllerAdvice("com.laker.notes")
@Slf4j
public class GlobalExceptionHandler {  
    /**
     * 验证bean类型
     */
    @ExceptionHandler(MethodArgumentNotValidException.class)
    public Response handleMethodArgumentNotValidException(MethodArgumentNotValidException e) {
        log.error(e.getMessage(), e);
        List<Map> result = new ArrayList<>();
        e.getBindingResult().getFieldErrors().forEach((fieldError) -> {
            result.add(Dict.create().set("field", fieldError.getField()).set("msg", fieldError.getDefaultMessage()));
        });
        return Response.error(result);
    }
    ...

验证:

在这里插入图片描述

2.简单平面参数验证

控制器:

控制类:加注解@Validated

@RestController
@Slf4j
@Validated
public class ValidatorController {    
	@RequestMapping("/validator")
    public Response validatorParam(@NotBlank(message = "姓名不能为空") String name, @Min(value = 0, message = "年龄应大于等于0")
    @Max(value = 150, message = "年龄应小于等于150") String age) {
        return Response.ok();
    }

验证:

在这里插入图片描述

3.手动验证

工具类:

public class ValidatorUtils {
    private static Validator validator;
    static {
        // 正常模式
        validator = Validation.byProvider(HibernateValidator.class)
                .configure()
                //快速故障模式。启用快速失败时,验证将在检测到第一个约束违规时停止
                .failFast(false)
                .buildValidatorFactory()
                .getValidator();
    }
    /**
     * 校验对象
     */
    public static <T> void validate(T object, Class<?>... groups) {
        Set<ConstraintViolation<T>> constraintViolationSet = validator.validate(object, groups);

        if (!constraintViolationSet.isEmpty()) {
            throw new ConstraintViolationException(constraintViolationSet);
        }
    }
}

控制器:

    @RequestMapping("/util")
    public Response validatorUtil(@RequestBody LakerUser lakerUser) {
        ValidatorUtils.validate(lakerUser);// 任意业务逻辑前后调用
        return Response.ok();
    }

验证

在这里插入图片描述

4.List验证

控制器:

方法入参:加@Valid,同时在控制层:添加@Validated注解

@RestController
@Slf4j
@Validated
public class ValidatorController { 
	@RequestMapping("/lakerusers")
    public Response validatorListBean(@RequestBody @Valid List<LakerUser> lakerUser) {
        return Response.ok();
    }

验证:

扩展

1.@Validated 和 @Valid 的异同

  • @Validated 是 Spring 实现的JSR-303@Valid的变体 ,支持验证组的规范。
  • @Valid 是JSR-303标准实现的校验注解。
注解范围嵌套校验组
@Validated可以标记类、方法、方法参数,不能用在成员属性(字段)上不支持支持
@Valid可以标记方法、构造函数、方法参数和成员属性(字段)上支持不支持

两者都可以用在方法入参上,但都无法单独提供嵌套验证功能,都能配合嵌套验证注解@Valid进行嵌套验证。

嵌套验证示例:

public class ClassRoom{
    @NotNull
    String name;
    @Valid  // 嵌套校验,校验参数内部的属性
    @NotNull
    Student student;
}
    @GetMapping("/room")   // 此处可使用 @Valid 或 @Validated, 将会进行嵌套校验
    public String validator(@Validated ClassRoom classRoom, BindingResult result) {
        if (result.hasErrors()) {
            return result.getFieldError().getDefaultMessage();
        }
        return "ok";
    }

2.快速故障模式

快速故障模式:启用快速失败时,验证将在检测到第一个约束违规时停止,默认是非快速故障模式

   // 正常模式
        validator = Validation.byProvider(HibernateValidator.class)
                .configure()
                //快速故障模式。启用快速失败时,验证将在检测到第一个约束违规时停止
                .failFast(true)
                .buildValidatorFactory()
                .getValidator();

3.验证相关注解和说明

验证注解验证的数据类型说明
@AssertFalseBoolean,boolean验证注解的元素值是false
@AssertTrueBoolean,boolean验证注解的元素值是true
@NotNull任意类型验证注解的元素值不是null
@Null任意类型验证注解的元素值是null
@Min(value=值)BigDecimal,BigInteger, byte,short, int, long,等任何Number或CharSequence(存储的是数字)子类型验证注解的元素值大于等于@Min指定的value值
@Max(value=值)和@Min要求一样验证注解的元素值小于等于@Max指定的value值
@DecimalMin(value=值)和@Min要求一样验证注解的元素值大于等于@ DecimalMin指定的value值
@DecimalMax(value=值)和@Min要求一样验证注解的元素值小于等于@ DecimalMax指定的value值
@Digits(integer=整数位数, fraction=小数位数)和@Min要求一样验证注解的元素值的整数位数和小数位数上限
@Size(min=下限, max=上限)字符串、Collection、Map、数组等验证注解的元素值的在min和max(包含)指定区间之内,如字符长度、集合大小
@Pastjava.util.Date,java.util.Calendar;Joda Time类库的日期类型验证注解的元素值(日期类型)比当前时间早
@Future与@Past要求一样验证注解的元素值(日期类型)比当前时间晚
@NotBlankCharSequence子类型验证注解的元素值不为空(不为null、去除首位空格后长度为0),不同于@NotEmpty,@NotBlank只应用于字符串且在比较时会去除字符串的首位空格
@Length(min=下限, max=上限)CharSequence子类型验证注解的元素值长度在min和max区间内
@NotEmptyCharSequence子类型、Collection、Map、数组验证注解的元素值不为null且不为空(字符串长度不为0、集合大小不为0)
@Range(min=最小值, max=最大值)BigDecimal,BigInteger,CharSequence, byte, short, int, long等原子类型和包装类型验证注解的元素值在最小值和最大值之间
@Email(regexp=正则表达式,flag=标志的模式)CharSequence子类型(如String)验证注解的元素值是Email,也可以通过regexp和flag指定自定义的email格式
@Pattern(regexp=正则表达式,flag=标志的模式)String,任何CharSequence的子类型验证注解的元素值与指定的正则表达式匹配

4.自定义验证注解

1.创建自定义注解

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD,ElementType.PARAMETER})
@Constraint(validatedBy = FlagValidatorClass.class) // 绑定对应校验器
public @interface FlagValidator {
    String[] value() default {};
 
    String message() default "flag is not found";
 
    Class<?>[] groups() default {};
 
    Class<? extends Payload>[] payload() default {};
}

2.编写对应校验器

/**
 * 标志位校验器
 */
public class FlagValidatorClass implements ConstraintValidator<FlagValidator, Integer> {
    private String[] values;
 
    /**
     * 初始化
     *
     * @param flagValidator 注解上设置的值
     */
    @Override
    public void initialize(FlagValidator flagValidator) {
        this.values = flagValidator.value();
    }
 
    /**
     * 校验
     *
     * @param value  被校验的值,即输入
     * @param constraintValidatorContext 校验上下文
     * @return 返回true证明校验通过,false校验失败
     */
    @Override
    public boolean isValid(Integer value, ConstraintValidatorContext constraintValidatorContext) {
        boolean isValid = false;
        // 当value为null,校验失败
        if (value == null) {
            return false;
        }
        //遍历校验
        for (int i = 0; i < values.length; i++) {
            if (values[i].equals(String.valueOf(value))) {
                isValid = true;
                break;
            }
        }
        return isValid;
    }
}

3.属性注解

    @NotNull(message = "标识位不能为空")
    @FlagValidator(value = {"0", "1"}, message = "标志位有误")
    private Integer flag;

参考:


🍎QQ群【837324215
🍎关注我的公众号【Java大厂面试官】,一起学习呗🍎🍎🍎

img