SpringBoot开发秘籍 - 集成参数校验及高阶技巧

Posted Javachichi

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了SpringBoot开发秘籍 - 集成参数校验及高阶技巧相关的知识,希望对你有一定的参考价值。

对于 web服务来说,为防止非法参数对业务造成影响,在
Controller层一定要对参数进行校验!本章我们以SpringBoot项目为例,介绍参数校验的基本用法以及一些高级技巧,希望能对你有所帮助。

简单使用

  1. 要在Springboot项目中加入参数校验功能首先得加入spring-boot-starter-validation依赖
`<dependency>  
 <groupId>org.springframework.boot</groupId>  
 <artifactId>spring-boot-starter-validation</artifactId>  
</dependency>  
`
  1. 然后给需要校验的字段添加上约束性注解,如我们对实体类参数进行校验
`@Data  
public class ValidEntity{  
    private int id;  
    @NotBlank  
    private String appId;  
    
    @NotBlank  
    private String name;  
    
    @Email  
    private String email;  
}  
`

常见约束注解如下:

注解功能
@AssertFalse可以为null,如果不为null的话必须为false
@AssertTrue可以为null,如果不为null的话必须为true
@DecimalMax设置不能超过最大值
@DecimalMin设置不能超过最小值
@Digits设置必须是数字且数字整数的位数和小数的位数必须在指定范围内
@Future日期必须在当前日期的未来
@Past日期必须在当前日期的过去
@Max最大不得超过此最大值
@Min最大不得小于此最小值
@NotNull不能为null,可以是空
@Null必须为null
@Pattern必须满足指定的正则表达式
@Size集合、数组、map等的size()值必须在指定范围内
@Email必须是email格式
@Length长度必须在指定范围内
@NotBlank字符串不能为null,字符串trim()后也不能等于“”
@NotEmpty不能为null,集合、数组、map等size()不能为0;字符串trim()后可以等于“”
@Range值必须在指定范围内
@URL必须是一个URL

注:此表格只是简单的对注解功能的说明,并没有对每一个注解的属性进行说明;可详见源码。

  1. 在Controller层对需要参数校验的方法加上@Validated注解

    参数校验一般分为两类:在Controller使用模型接收数据时, @Validated注解直接放在该模型参数前即可。

`@PostMapping(value = "test1")  
public String test1(@Validated @RequestBody ValidEntity validEntity){  
 return "test1 valid success";  
}  
  
@PostMapping(value = "test3")  
public String test3(@Validated ValidEntity validEntity){  
 return "test3 valid success";  
}  
`

当我们是直接在Controller层中的参数前,使用约束注解时,@Validated要直接放在类上

`@PostMapping(value = "test2")  
public String test2(@Email String email){  
    return "test2 valid success";  
}  
`

此时需要在主类上增加@Validated注解

`@Validated  
@RestController  
@RequestMapping("/demo/valid")  
public class ValidController {  
 ...  
}  
`

在参数校验时我们既可以使用@Validated也可以使用@Valid注解,两者功能大部分类似;

主要区别在于:

@Valid属于javax下的,而@Validated属于spring下;

@Valid支持嵌套校验、而@Validated不支持,@Validated支持分组,而@Valid不支持。

统一异常处理

如果参数校验未通过Spring会抛出三种类型的异常

  1. 当对@RequestBody需要的参数进行校验时会出现org.springframework.web.bind.MethodArgumentNotValidException

  1. 当直接校验具体参数时会出现javax.validation.ConstraintViolationException,也属于ValidationException异常

  1. 当直接校验对象时会出现org.springframework.validation.BindException

在SpringBoot中统一拦截处理只需要在配置类上添加 @RestControllerAdvice注解,然后在具体方法中通过 @ExceptionHandler指定需要处理的异常,具体代码如下:

`@RestControllerAdvice  
@Slf4j  
public class GlobalExceptionHandler {  
    public static final String ERROR_MSG = "系统异常,请联系管理员。";  
  
    @ExceptionHandler(value = {BindException.class, ValidationException.class, MethodArgumentNotValidException.class})  
    public ResponseEntity<Result<String>> handleValidatedException(Exception e) {  
        Result<String> resp = null;  
  
        if (e instanceof MethodArgumentNotValidException) {  
            // BeanValidation exception  
            MethodArgumentNotValidException ex = (MethodArgumentNotValidException) e;  
            resp = new Result<>(Integer.toString(HttpStatus.BAD_REQUEST.value()),  
                    ex.getBindingResult().getAllErrors().stream().map(ObjectError::getDefaultMessage).collect(Collectors.joining(", "))  
                    , getStackTrace(ex));  
        } else if (e instanceof ConstraintViolationException) {  
            // BeanValidation GET simple param  
            ConstraintViolationException ex = (ConstraintViolationException) e;  
            resp = new Result<>(Integer.toString(HttpStatus.BAD_REQUEST.value()),  
                    ex.getConstraintViolations().stream().map(ConstraintViolation::getMessage).collect(Collectors.joining(", "))  
                    , getStackTrace(ex));  
        } else if (e instanceof BindException) {  
            // BeanValidation GET object param  
            BindException ex = (BindException) e;  
            resp = new Result<>(Integer.toString(HttpStatus.BAD_REQUEST.value()),  
                    ex.getAllErrors().stream().map(ObjectError::getDefaultMessage).collect(Collectors.joining(", "))  
                    , getStackTrace(ex));  
        }  
  
        return new ResponseEntity<>(resp,HttpStatus.BAD_REQUEST);  
    }  
  
  
    private String getStackTrace(Exception e) {  
        //打印日志开关,可通过配置读取  
        boolean printStrackTrace = false;  
        if(printStrackTrace){  
            StringWriter sw = new StringWriter();  
            e.printStackTrace(new PrintWriter(sw));  
            return sw.toString();  
        }else{  
            return ERROR_MSG;  
        }  
  
    }  
  
}`

最终实现效果如下:

参数分组

有下面一个实体类,我们需要对其进行参数校验。

`@Data  
public class ValidEntity {  
    private int id;  
  
    @NotBlank  
    private String appId;  
  
    @NotBlank  
    private String name;  
  
    @Email  
    private String email;  
}  
`

但是实际业务是在编辑的时候 appId才是必填,在新增的时候 name必填,这时候可以用groups分组功能来实现:同一个模型在不同场景下,动态区分校验模型中的不同字段。

使用方式

  1. 首先我们定义一个分组接口ValidGroup,再在分组接口总定义出多个不同的操作类型,Create,Update,Query,Delete
`public interface ValidGroup extends Default{  
    
    interface Crud extends ValidGroup{  
        
        interface Create extends Crud{  
  
        }  
      
        interface Update extends Crud{  
  
        }  
        
        interface Query extends Crud{  
  
        }  
    
        interface Delete extends Crud{  
  
        }  
    }  
}  
`

这里的 ValidGroup继承了Default,当然也可以不继承,具体区别我们后面再说。

  1. 在模型中给校验参数分配分组
`@Data  
@ApiModel(value="ValidEntity")  
public class ValidEntity {  
    private int id;  
  
    @NotBlank(groups = ValidGroup.Crud.Update.class)  
    private String appId;  
  
    @NotBlank(groups = ValidGroup.Crud.Create.class)  
    private String name;  
  
    @Email  
    private String email;  
}  
`

tips:这里@Email注解未指定分组,默认会属于Default分组,appId和name指定了分组就不会再属于Default分组了。

  1. 在参数校验时通过value属性指定分组

这里通过 @Validated(value = ValidGroup.Crud.Update.class)指定了具体的分组,上面提到的是否继承Default的区别在于:

  • 如果继承了Default,@Validated标注的注解也会校验未指定分组或者Default分组的参数,比如email

  • 如果不继承Default则不会校验未指定分组的参数,需要加上@Validated(value = {ValidGroup.Crud.Update.class, Default.class}才会校验

快速失败(Fali Fast)

默认情况下在对参数进行校验时Spring Validation会校验完所有字段然后才抛出异常,可以通过配置开启 Fali Fast模式,一旦校验失败就立即返回。

`@Configuration  
public class ValidatedConfig {  
  
    @Bean  
    public Validator validator() {  
        ValidatorFactory validatorFactory = Validation.byProvider(HibernateValidator.class)  
                .configure()  
                // 快速失败模式  
                .failFast(true)  
                .buildValidatorFactory();  
        return validatorFactory.getValidator();  
    }  
}  
`

以上,希望对你有所帮助!

最后

以下是Java面试1—到5年以上开发必问到的面试问点,也都是一线互联网公司Java面试必备技能,下面是参照阿里年薪50W所需具备的技能图,大家可以参考下!
在这里插入图片描述

同时针对这12个技能,我在这整理了一份Java架构进阶面试专题PDF文档(含450题解析,包括Dubbo、Redis、Netty、zookeeper、Spring cloud、分布式、高并发,设计模式,MySQL等知识点解析,内容丰富,图文结合!)

蚂蚁金服Java研发岗三面:MySQL+秒杀+Redis+JVM等(终获offer)
这份专题文档是免费分享的,有需要的朋友可以看向下面来获取!!

需要完整版文档的小伙伴,可以一键三连,下方获取免费领取方式!
在这里插入图片描述

以上是关于SpringBoot开发秘籍 - 集成参数校验及高阶技巧的主要内容,如果未能解决你的问题,请参考以下文章

SpringBoot 开发秘籍 - 整合参数校验

SpringBoot - 优雅的实现参数校验高级进阶

SpringBoot中集成参数校验

SpringBoot中集成参数校验

华为12年开发大老,熬夜2个月整理的SpringBoot手册,绝对的涨薪秘籍

SpringBoot 如何进行参数校验,老鸟们都这么玩的!