zoukankan      html  css  js  c++  java
  • SpringBoot,SpringMvc 参数校验 用法详解

    1.情景展示

    在实际开发过程中,针对前端请求参数的校验是一个不小的工作量。

    什么时候需要对请求参数进行校验?

    情形1:前后端分离

    前后端分离,虽然会提高项目的开发进度,但同样也存在前后端开发人员交流不及时等问题。

    比方说:性别参数,后端要求只能传1或2,前端非要给你传男或女,当前后端对于数据的要求标准不一致时,就会出现问题。

    后台对入参进行校验,一方面,可以提高数据的规范性;另一方面,也可以增加数据的安全性(比如:数据在传输过程中被篡改)。

    情形2:对外提供接口

    本质上,前后端分离,后台提供请求,也是属于接口,这里特指的是后台对后台。

    也就是说,别的项目或者公司需要调用咱们写的接口,这个时候,参数的校验就显得格外重要,不想前后端那种,后台加不加校验都没有太大的影响。

    2.准备工作

    关于校验标准,可供java使用的一共有两套:

    一种是:Java API规范 (JSR303) 定义了Bean校验的标准validation-api,但没有提供实现。

    另一种是:hibernate validation是对这个规范的实现,并增加了校验注解如@Email@Length等。

    Spring Validation是对hibernate validation的二次封装,用于支持spring mvc参数自动校验。

    接下来,我们以spring-boot项目为例,介绍Spring Validation的使用。

    关于jar包的引用

    如果spring-boot版本小于2.3.xspring-boot-starter-web会自动传入hibernate-validator依赖;

    如果spring-boot版本大于2.3.x,则需要手动引入依赖:

    <!--spring对参数进行校验:hibernate validator-->
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-validator</artifactId>
        <version>6.0.18.Final</version>
    </dependency>

    现在网络上都是旧的内容,用的是:

     

    完全没有必要,使用第一个jar包就可以了。

    @Valid和@Validated的区别

    3.具体实现

    用法1:requestParam参数校验

    描述:通常用于get请求或者请求参数比较少的情形。

    校验生效的前提:

    必须在Controller类上标注@Validated注解,在方法或者参数前添加无效!

    如果校验失败,会抛出ConstraintViolationException异常。

    用法:

    将请求入参一一在请求方法()内,进行罗列,并将校验注解添加在对应入参的前面。

     

    用法2:pathVariable参数校验

    描述:通过{}来动态配置请求路径,并将请求路径当成方法的入参之一。

    校验生效的前提:

    必须在Controller类上标注@Validated注解,在方法或者参数前添加无效!

    如果校验失败,会抛出ConstraintViolationException异常。

    用法:校验注解可以放在@PathVariable前面也可以放在它的后面。

    用法3:responseBody参数校验(application/json)

    当请求方法入参有@RequestBody注解的时候,spring会将它识别成JSON格式的请求,要求调用方必须发送application/json格式的数据;

    第1步:在实体类的字段上加上约束注解;

    第2步:在方法参数上声明校验注解。

    格式:@RequestBody+@Validated+实体类

    或者:@RequestBody+@Valid+实体类

    这种情况下,使用@Valid@Validated都可以(只能用@Valid或@Validated的地方,下面会讲)

    用法4:responseBody参数校验(application/x-www-form-urlencoded)

    当请求方法入参只有实体类接收的时候,spring会将它识别成FORM表单请求,要求调用方必须发送application/x-www-form-urlencoded格式的数据;

    第1步:在实体类的字段上加上约束注解;

    第2步:在方法参数上声明校验注解。

    同样地,使用@Valid@Validated都可以

    4.参数校验配置

    Spring Validation默认会校验完所有字段,然后才抛出异常。可以通过一些简单的配置,开启Fali Fast模式,一旦校验失败就立即返回。

    查看代码
    /**
     * 请求参数校验配置
     * @description:
     * @author: Marydon
     * @date: 2020年08月10日 0010 16:47
     */
    @Configuration
    public class WebParamValidateConfig {
        @Bean
        public Validator validator() {
            ValidatorFactory validatorFactory = Validation.byProvider(HibernateValidator.class)
                    .configure()
                    //failFast的意思只要出现校验失败的情况,就立即结束校验,不再进行后续的校验。
                    .failFast(true)
                    .buildValidatorFactory();
            return validatorFactory.getValidator();
        }
    
        @Bean
        public MethodValidationPostProcessor methodValidationPostProcessor() {
            MethodValidationPostProcessor methodValidationPostProcessor = new MethodValidationPostProcessor();
            methodValidationPostProcessor.setValidator(validator());
            return methodValidationPostProcessor;
        }
    }

    5.统一异常处理

    如果校验失败,会抛出异常,我们需要对异常进行管理,以便返回一个更友好的提示。

    下面是我总结的异常拦截配置类:

    查看代码
    import org.jetbrains.annotations.NotNull;
    import org.springframework.context.support.DefaultMessageSourceResolvable;
    import org.springframework.http.converter.HttpMessageNotReadableException;
    import org.springframework.validation.BindException;
    import org.springframework.web.HttpMediaTypeNotSupportedException;
    import org.springframework.web.HttpRequestMethodNotSupportedException;
    import org.springframework.web.bind.MethodArgumentNotValidException;
    import org.springframework.web.bind.annotation.ExceptionHandler;
    import org.springframework.web.bind.annotation.RestControllerAdvice;
    
    import javax.validation.ConstraintViolation;
    import javax.validation.ConstraintViolationException;
    import java.util.stream.Collectors;
    
    
    /**
     * 全局异常处理器
     * @description: @ControllerAdvice默认会对所有的请求进行处理;
     * 对请求入参的校验异常,根本进不到Controller的方法体内,所以,只能在这拦截异常后返回友好错误提示
     * 加basePackages,可以只对具体包名下面的请求进行处理
     * @ControllerAdvice是一个增强的 Controller。使用这个 Controller,可以实现三个方面的功能:
     * 全局异常处理
     * 全局数据绑定
     * 全局数据预处理
     * 只拦截Controller,不会拦截Interceptor的异常
     * @author: Marydon
     * @date: 2020年08月10日 0010 16:39
     */
    // 异常拦截位置
    @RestControllerAdvice(basePackages = {"com.xx"
                                            ,"com.yy"})
    public class CzWebExceptionHandler {
    
        //处理Get请求中 使用@Valid 验证路径中请求实体校验失败后抛出的异常,详情继续往下看代码
        @ExceptionHandler(BindException.class)
        public CzResponseDto<JSONObject> BindExceptionHandler(@NotNull BindException e) {
            String errorMsg = e.getBindingResult().getAllErrors().stream().map(DefaultMessageSourceResolvable::getDefaultMessage).collect(Collectors.joining());
            return CzResult.error("接口调取失败:" + errorMsg, null);
        }
    
        //处理请求参数格式错误 @RequestParam上validate失败后抛出的异常是javax.validation.ConstraintViolationException
        @ExceptionHandler(ConstraintViolationException.class)
        public CzResponseDto<JSONObject> ConstraintViolationExceptionHandler(@NotNull ConstraintViolationException e) {
            String errorMsg = e.getConstraintViolations().stream().map(ConstraintViolation::getMessage).collect(Collectors.joining());
            return CzResult.error("接口调取失败:" + errorMsg, null);
        }
    
        //处理请求参数格式错误 @RequestBody上validate失败后抛出的异常是MethodArgumentNotValidException异常。
        @ExceptionHandler(MethodArgumentNotValidException.class)
        public CzResponseDto<JSONObject> MethodArgumentNotValidExceptionHandler(@NotNull MethodArgumentNotValidException e) {
            String errorMsg = e.getBindingResult().getAllErrors().stream().map(DefaultMessageSourceResolvable::getDefaultMessage).collect(Collectors.joining());
            return CzResult.error("接口调取失败:" + errorMsg, null);
        }
    
        // 要求Content-type为application/json,但是内容类型却是text/plain或者text时会被该异常捕获
        @ExceptionHandler(HttpMediaTypeNotSupportedException.class)
        public CzResponseDto<JSONObject> HttpMediaTypeNotSupportedExceptionHandler(@NotNull HttpMediaTypeNotSupportedException e) {
            String errorMsg = "本接口不接收application/json以外格式的数据,请检查入参是否是标准的json数据!\n" + e.getMessage();
            return CzResult.error("接口调取失败:" + errorMsg, null);
        }
    
        @ExceptionHandler(Exception.class)
        public CzResponseDto<JSONObject> handleException(@NotNull Exception e) {
            String errorMsg = "系统异常,请联系开发人员Marydon进行排错!\n" + e.getMessage();
            return CzResult.error("接口调取失败:" + errorMsg, null);
        }
    
        // 运行异常
        @ExceptionHandler(RuntimeException.class)
        public CzResponseDto<JSONObject> handleRuntimeException(@NotNull Exception e) {
            return CzResult.error("接口调取失败:" + e.getMessage(), null);
        }
    
        // 服务异常
        @ExceptionHandler(ServiceException.class)
        public CzResponseDto<JSONObject> handleServiceException(@NotNull Exception e) {
            return CzResult.error("接口调取失败:" + e.getMessage(), null);
        }
    
        // 请求方式异常(仅支持post请求)
        @ExceptionHandler(HttpRequestMethodNotSupportedException.class)
        public CzResponseDto<JSONObject> handleHttpRequestMethodNotSupportedException(@NotNull Exception e) {
            return CzResult.error("接口调取失败:" + e.getMessage(), null);
        }
    
        // 不存在的请求方法InterfaceMethod,在CzRequestParams可以查看支持的接口
        // 注意:当请求数据格式为非text或text/plain时,也会抛出该异常(application/javascript,application/xml,text/xml,text/html)
        // 无请求入参时也会抛出该异常
        @ExceptionHandler(HttpMessageNotReadableException.class)
        public CzResponseDto<JSONObject> handleHttpMessageNotReadableException(@NotNull Exception e) {
            // return CzResult.error("接口调取失败:未知的接口方法,请仔细核对入参InterfaceMethod的值!", null);
            return CzResult.error("接口调取失败:" + e.getMessage(), null);
        }
    }
    

    仅供参考。

    6.常用注解

    @NotBlank:只用在String上,表示:传进来的值不能为null,而且调用trim()后,长度必须大于0,即:必须有实际字符;

    @NotNull:不能为null,但值可以为empty(分配了内存空间),只能校验String类型和对象;

    不能校验:八大基本数据类型(因为基本数据类型有默认值:byte,short,int,long,double,flot,char,boolean);

    即该参数是必传项,但其值可以为空。

    如果非得用基础数据类型接收的话

    那就只能和基本数据类型的默认值比较,进行判断啦。

    或者:把基本数据类型改成String,然后,在需要的时候,再将其进行数据类型转换,转成自己所需的数据类型。

    @NotEmpty:不能为null,而且长度必须大于0,只能校验字符串;

    @Max:最大值,限制该参数的最大值;

    @Min:最小值,限制该参数的最小值;

    说明:

    @Max和@Min只能校验数值类型,也就是说,限制该参数的数据类型只能为数字!

    这两个注解,通常一块使用,可以单独使用;

    用于接收的数据类型既可以是数值类型,也可以是字符串类型。

    @Length:校验字符串长度;

    @Size:校验数组、集合大小(java,经测试无效);

    @Pattern:正则表达式校验

    使用标准的正则表达式用法,带反斜杠\的会自动转义。

    常用正则表达式:

    固定字符:regexp = "^(门诊|住院|资往)$";

    可以为空或8个数字:regexp = "^(\s{0}|\d{8})$";

    长度必须为6的字符串:regexp = "^([0-9a-z]{6})$";

    可以为空或者正整数[1,99]:regexp = "^(\s{0}|[1-9]\d{0,1})$";

    校验手机号:regexp = "^1(3[0-9]|4[5,7]|5[0,1,2,3,5,6,7,8,9]|6[2,5,6,7]|7[0,1,7,8]|8[0-9]|9[1,8,9])\d{8}$"

    身份证号校验:regexp = "^[1-9]\d{5}(18|19|20)\d{2}((0[1-9])|(1[0-2]))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$"

    5.扩展延伸

    延伸1:分组校验

    在实际项目中,可能多个方法需要使用同一个实体类来接收参数,而不同方法的校验规则很可能是不一样的;

    这个时候,简单地在实体的字段上加约束注解无法解决这个问题。

    因此,spring-validation支持了分组校验的功能,专门用来解决这类问题。

    举个栗子:

    A方法要求参数1的值必须为1,B方法则要求其值必须为2,如何实现?

    第1步:在实体类当中添加接口类;

    把接口类用作分组的依据;

    注意:在实体类当中添加的接口,没有实际意义,仅仅将其作为分组依据;

    由于spring校验的groups只能这样用,没有办法。

    第1步:在约束注解里,添加该注解生效的的分组信息groups。

    多个组之间使用逗号隔开;

    并且,组必须以".class"进行结尾。

    针对不同的组,可以添加不同的校验规则。

    第2步:@Validated注解上指定校验分组。

    注意:分组校验只能使用注解@Validated,不能使用@Valid!

    另外,方法上使用了分组校验,实体类需要多组共用的字段规则校验,也必须添加组,即使是:校验规则一致的。

    否则的话,该校验规则将会失效。

    延伸2:嵌套校验

    当入参实体类的某字段也是对象时,这时,需要对该对象里的字段进行校验时,这就牵扯到了:嵌套校验;

    此时,入参实体类的对应的字段对象,必须标记@Valid注解。

    嵌套的实体类,示例:

    延伸3:集合校验(list)

    如果请求体直接传递了json数组给后台,并希望对数组中的每一项都进行参数校验。

    此时,如果我们直接使用java.util.Collection下的list或者set来接收数据,参数校验并不会生效!

    我们可以使用自定义list集合来接收参数:

    第1步:包装List类型,并声明@Valid注解;

    查看代码
    public class ValidationList<E> implements List<E> {
        @Delegate
        @Valid
        public List<E> list = new ArrayList<>();
    
        @Override
        public String toString() {
            return list.toString();
        }
    }

    @Delegate注解受lombok版本限制,1.18.6以上版本可支持;

    如果校验不通过,会抛出NotReadablePropertyException,同样可以使用统一异常进行处理。

    第2步:校验调用

    格式:@Validated + ValidationList<实体类>。

    比如,我们需要一次性保存多个User对象,Controller层的方法可以这么写:

    查看代码
    @PostMapping("/saveList")
    public Result saveList(@RequestBody @Validated(UserDTO.Save.class) ValidationList<UserDTO> userList) {
    
        return Result.ok();
    }

    延伸4:自定义校验

    延伸5:编程式校验

     

    写在最后

      哪位大佬如若发现文章存在纰漏之处或需要补充更多内容,欢迎留言!!!

     相关推荐:

  • 相关阅读:
    离散数学随笔2
    离散数学随笔1
    java多线程实现线程同步
    c语言细节
    堆的简单实现和应用
    快速排序分析
    ORACLE PRAGMA AUTONOMOUS_TRANSACTION 自治事务 单独提交某一段操作
    System.out.println() 输出 快捷键
    最全最新🇨🇳中国【省、市、区县、乡镇街道】json,csv,sql数据
    使用 js 设置组合快捷键,支持多个组合键定义,还支持 React
  • 原文地址:https://www.cnblogs.com/Marydon20170307/p/15707793.html
Copyright © 2011-2022 走看看