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

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

[[400097]]

本文转载自微信公众号「JAVA日知录」,作者单一色调。转载本文请联系JAVA日知录公众号。

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

简单使用

1.要在Springboot项目中加入参数校验功能首先得加入spring-boot-starter-validation依赖

  1. <dependency> 
  2.  <groupId>org.springframework.boot</groupId> 
  3.  <artifactId>spring-boot-starter-validation</artifactId> 
  4. </dependency> 

2.然后给需要校验的字段添加上约束性注解,如我们对实体类参数进行校验

  1. @Data 
  2. public class ValidEntity{ 
  3.     private int id; 
  4.     @NotBlank 
  5.     private String appId; 
  6.    
  7.     @NotBlank 
  8.     private String name
  9.    
  10.     @Email 
  11.     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

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

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

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

  1. @PostMapping(value = "test1"
  2. public String test1(@Validated @RequestBody ValidEntity validEntity){ 
  3.  return "test1 valid success"
  4.  
  5. @PostMapping(value = "test3"
  6. public String test3(@Validated ValidEntity validEntity){ 
  7.  return "test3 valid success"

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

  1. @PostMapping(value = "test2"
  2. public String test2(@Email String email){ 
  3.     return "test2 valid success"

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

  1. @Validated 
  2. @RestController 
  3. @RequestMapping("/demo/valid"
  4. public class ValidController { 
  5.  ... 

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

主要区别在于:

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

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

统一异常处理

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

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

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

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

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

  1. @RestControllerAdvice 
  2. @Slf4j 
  3. public class GlobalExceptionHandler { 
  4.     public static final String ERROR_MSG = "系统异常,请联系管理员。"
  5.  
  6.     @ExceptionHandler(value = {BindException.class, ValidationException.class, MethodArgumentNotValidException.class}) 
  7.     public ResponseEntity<Result<String>> handleValidatedException(Exception e) { 
  8.         Result<String> resp = null
  9.  
  10.         if (e instanceof MethodArgumentNotValidException) { 
  11.             // BeanValidation exception 
  12.             MethodArgumentNotValidException ex = (MethodArgumentNotValidException) e; 
  13.             resp = new Result<>(Integer.toString(HttpStatus.BAD_REQUEST.value()), 
  14.                     ex.getBindingResult().getAllErrors().stream().map(ObjectError::getDefaultMessage).collect(Collectors.joining(", ")) 
  15.                     , getStackTrace(ex)); 
  16.         } else if (e instanceof ConstraintViolationException) { 
  17.             // BeanValidation GET simple param 
  18.             ConstraintViolationException ex = (ConstraintViolationException) e; 
  19.             resp = new Result<>(Integer.toString(HttpStatus.BAD_REQUEST.value()), 
  20.                     ex.getConstraintViolations().stream().map(ConstraintViolation::getMessage).collect(Collectors.joining(", ")) 
  21.                     , getStackTrace(ex)); 
  22.         } else if (e instanceof BindException) { 
  23.             // BeanValidation GET object param 
  24.             BindException ex = (BindException) e; 
  25.             resp = new Result<>(Integer.toString(HttpStatus.BAD_REQUEST.value()), 
  26.                     ex.getAllErrors().stream().map(ObjectError::getDefaultMessage).collect(Collectors.joining(", ")) 
  27.                     , getStackTrace(ex)); 
  28.         } 
  29.  
  30.         return new ResponseEntity<>(resp,HttpStatus.BAD_REQUEST); 
  31.     } 
  32.  
  33.  
  34.     private String getStackTrace(Exception e) { 
  35.         //打印日志开关,可通过配置读取 
  36.         boolean printStrackTrace = false
  37.         if(printStrackTrace){ 
  38.             StringWriter sw = new StringWriter(); 
  39.             e.printStackTrace(new PrintWriter(sw)); 
  40.             return sw.toString(); 
  41.         }else
  42.             return ERROR_MSG; 
  43.         } 
  44.  
  45.     } 
  46.  

最终实现效果如下:

参数分组

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

  1. @Data 
  2. public class ValidEntity { 
  3.     private int id; 
  4.  
  5.     @NotBlank 
  6.     private String appId; 
  7.  
  8.     @NotBlank 
  9.     private String name
  10.  
  11.     @Email 
  12.     private String email; 

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

使用方式

首先我们定义一个分组接口ValidGroup,再在分组接口总定义出多个不同的操作类型,Create,Update,Query,Delete

  1. public interface ValidGroup extends Default
  2.    
  3.     interface Crud extends ValidGroup{ 
  4.        
  5.         interface Create extends Crud{ 
  6.  
  7.         } 
  8.      
  9.         interface Update extends Crud{ 
  10.  
  11.         } 
  12.        
  13.         interface Query extends Crud{ 
  14.  
  15.         } 
  16.    
  17.         interface Delete extends Crud{ 
  18.  
  19.         } 
  20.     } 

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

在模型中给校验参数分配分组

  1. @Data 
  2. @ApiModel(value="ValidEntity"
  3. public class ValidEntity { 
  4.     private int id; 
  5.  
  6.     @NotBlank(groups = ValidGroup.Crud.Update.class) 
  7.     private String appId; 
  8.  
  9.     @NotBlank(groups = ValidGroup.Crud.Create.class) 
  10.     private String name
  11.  
  12.     @Email 
  13.     private String email; 

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

在参数校验时通过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模式,一旦校验失败就立即返回。

  1. @Configuration 
  2. public class ValidatedConfig { 
  3.  
  4.     @Bean 
  5.     public Validator validator() { 
  6.         ValidatorFactory validatorFactory = Validation.byProvider(HibernateValidator.class) 
  7.                 .configure() 
  8.                 // 快速失败模式 
  9.                 .failFast(true
  10.                 .buildValidatorFactory(); 
  11.         return validatorFactory.getValidator(); 
  12.     } 

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

 

责任编辑:武晓燕 来源: JAVA日知录
相关推荐

2021-04-21 09:04:43

开发SpringBoot框架

2023-11-08 13:33:00

AOP技术信息

2023-03-16 08:23:33

2022-12-30 08:49:41

SpringBoot@Validated

2023-11-29 07:23:04

参数springboto

2022-05-03 10:43:43

SpringJava

2021-08-12 10:32:50

Spring Boot参数校验分组校验

2022-11-10 07:53:54

Spring参数校验

2023-10-18 07:36:56

SpringBootJava

2021-08-10 15:11:27

Spring Boot参数校验

2021-11-10 10:03:18

SpringBootJava代码

2011-04-29 14:08:19

手机社交游戏游戏开发

2023-03-31 08:10:50

2010-08-02 09:21:48

Flex模块化

2022-09-01 06:54:28

CSS前端

2022-12-22 08:34:22

CSS不规则图形

2011-02-22 14:07:52

2020-10-26 15:01:02

Spring Boot源码参数

2024-01-30 10:11:00

SpringBoot项目开发

2017-03-02 13:32:36

Android开发开发者
点赞
收藏

51CTO技术栈公众号