springboot参数校验

springboot参数传递
  • @PathVariable
  • @RequestParam
  • @RequestBody
JSR303

jsr303 : 也称 bean validation 规范,用于java bean 验证的标准API,,他定义了一组注解,可以在javabean 的属性上声明验证规则

JSR: java specification request

常用注解:

  • @NotNull
  • @Size : 定义属性的最小和最大长度
  • @Min
  • @Max
  • @Pattern : 属性值必须匹配正则
  • @Email
  • @Valid : 对嵌套对象或者集合触发验证
springboot中校验注解

在类上打上 @Validated 才会开启校验

校验javabean 也要设置 @Validated ,,javabean中的每一个字段,也需要验证,,字段可能是一个普通字段,也可能是对象,,如果是对象,,使用@Valid 对嵌套对象触发验证

@Valid@Validated 区别: 这两个都可以对javabean 开启验证,,@Valid是java提供的,,@Validated 是spring 对@Valid的扩展,,一般开启验证使用 @Validated,指定嵌套验证使用@Valid

自定义校验注解

自定义一个注解,校验两次输入密码一样:

  • 这个注解需要遵循一定的规范,他有两个模板方法,必须要有
java 复制代码
@Documented // 注解里面的注释加入到文档
 @Retention(RetentionPolicy.RUNTIME)
 @Target({ElementType.TYPE})
@Constraint(validatedBy = PasswordValidator.class)  // 指定多个校验类
public @interface PasswordEquals {
 int min() default 3;
 int max() default 6;
 String message() default "password not equal";

 /**
  *  如果要自定义校验  有两个模板方法,必须加上,,规范必须要有这两个东西
  */
 Class<?>[] groups() default {};
 Class<? extends Payload>[] payload() default {};
}
  • 写你自己的验证逻辑,实现ConstraintValidator接口,,这个接口第一个泛型是:你自定义的注解,, 第二个泛型是: 自定义注解要打在什么类型的数据上,,如果打在类上,就是类的类型,如果打在字段上,就是字段的类型
    这里面可以获取注解传入的值,,和javabean传入的值,,在isValid方法中进行比较,返回true,则验证通过
java 复制代码
/**
 * 泛型:
 *      <自定义注解的类型,自定义注解修饰的目标的类型>
 *
 *
 *     如果注解在类上,,修饰的是类
 *     如果注解在字段上,,修饰的是字段 ,,就是字段的类型
 */
public class PasswordValidator implements ConstraintValidator<PasswordEquals, PersonDTO> {

    private int min;
    private int max;

    /**
     * 校验通过 返回true,不通过返回false
     */
    @Override
    public boolean isValid(PersonDTO personDTO, ConstraintValidatorContext context) {
       // 没有考虑密码为空
        String password01 = personDTO.getPassword01();
        String password02 = personDTO.getPassword02();

        return password01.equals(password02);
    }

    /**
     * 获取注解传入的值
     */
    @Override
    public void initialize(PasswordEquals constraintAnnotation) {
        this.min = constraintAnnotation.min();
        this.max = constraintAnnotation.max();
        ConstraintValidator.super.initialize(constraintAnnotation);
    }
}
  • 将自定义的 ContraintValidator 绑定到指定注解上
    一个注解可以绑定多个 ConstraintValidator 指定多个校验类
参数校验的异常
  • ConstraintViolationException
    路径中传递参数报错 ,比如 @PathVarible ,,@RequestParam 传递的路径参数和query参数报错
    这个异常: 系统会自动拼接这些参数的错误,,在异常的message中返回
java 复制代码
 @ExceptionHandler(ConstraintViolationException.class)
    @ResponseStatus(HttpStatus.BAD_REQUEST)
    public UnifyResponse handleConstraintViolationException(HttpServletRequest request,ConstraintViolationException e){
        System.out.println("e = " + e);
        // 所有的验证
        Set<ConstraintViolation<?>> constraintViolations = e.getConstraintViolations();

        // 在 异常的 message 中拼接好了,,如果是在自定义异常要定制化显示,,就需要遍历 ConstraintViolation自己拼接
        return new UnifyResponse(10001, e.getMessage(), request.getMethod() + " " + request.getRequestURI());

    }
  • MethodArgumentNotValidException
    在对javabean验证的时候报错,,,多个校验器没有通过会返回一个集合,,需要自己拼接错误信息,所有异常都要处理,,不能只处理一个
java 复制代码
 @ExceptionHandler(MethodArgumentNotValidException.class)
    @ResponseStatus(HttpStatus.BAD_REQUEST) // 400 参数错误
    public UnifyResponse handleBeanValidation(HttpServletRequest request,MethodArgumentNotValidException e){
        String uri = request.getRequestURI();
        String method = request.getMethod();
        // 多个校验器都没有通过,,返回集合
        List<ObjectError> allErrors = e.getBindingResult().getAllErrors();

        Optional<String> errorMsg = allErrors.stream().map(DefaultMessageSourceResolvable::getDefaultMessage).reduce((a, b) -> (a + "," + b ));
        System.out.println("errormsg = " + errorMsg);


        return new UnifyResponse(10001, errorMsg.orElse(""), method + "  " + uri);
    }
相关推荐
手握风云-5 小时前
JavaEE 进阶第五期:Maven 之道,项目的依赖艺术与构建哲学
java·java-ee·maven
郝开6 小时前
Spring Boot 2.7.18(最终 2.x 系列版本):版本概览;兼容性与支持;升级建议;脚手架工程搭建
java·spring boot·后端
2301_796512527 小时前
Rust编程学习 - 如何利用代数类型系统做错误处理的另外一大好处是可组合性(composability)
java·学习·rust
A.说学逗唱的Coke7 小时前
【观察者模式】深入 Spring 事件驱动模型:从入门到微服务整合实战
spring·观察者模式·微服务
清水7 小时前
Spring Boot企业级开发入门
java·spring boot·后端
lzjava20247 小时前
Spring AI使用知识库增强对话功能
人工智能·python·spring
一个不称职的程序猿7 小时前
高并发场景下的缓存利器
java·缓存
Q_Q5110082857 小时前
python+django/flask的校园活动中心场地预约系统
spring boot·python·django·flask·node.js·php
2301_801252227 小时前
Tomcat的基本使用作用
java·tomcat
lkbhua莱克瓦247 小时前
Java基础——常用算法3
java·数据结构·笔记·算法·github·排序算法·学习方法