• Spring Boot 实现参数校验基本使用


    简单使用

    Java API规范 (JSR303) 定义了Bean校验的标准validation-api,但没有提供实现。hibernate validation是对这个规范的实现,并增加了校验注解如@Email@Length等。Spring Validation是对hibernate validation的二次封装,用于支持spring mvc参数自动校验。接下来,我们以spring-boot项目为例,介绍Spring Validation的使用。

    引入依赖

    如果spring-boot版本小于2.3.xspring-boot-starter-web会自动传入hibernate-validator依赖。如果spring-boot版本大于2.3.x,则需要手动引入依赖:

    1. <dependency>
    2. <groupId>org.springframework.bootgroupId>
    3. <artifactId>spring-boot-starter-validationartifactId>
    4. <version>2.6.4version>
    5. dependency>

    对于web服务来说,为防止非法参数对业务造成影响,在Controller层一定要做参数校验的!大部分情况下,请求参数分为如下两种形式:

    1. POSTPUT请求,使用requestBody传递参数;

    2. GET请求,使用requestParam/PathVariable传递参数。

    下面我们简单介绍下requestBodyrequestParam/PathVariable的参数校验实战!

    requestBody参数校验

    POSTPUT请求一般会使用requestBody传递参数,这种情况下,后端使用DTO 对象进行接收。只要给 DTO 对象加上@Validated注解就能实现自动参数校验。比如,有一个保存User的接口,要求userName长度是2-10password字段长度是6-20。如果校验失败,会抛出MethodArgumentNotValidException异常,Spring默认会将其转为400(Bad Request)请求。

    • DTO字段上声明约束注解

    1. @Data  
    2. public class UserDTO {  
    3.   
    4.     private Long userId;  
    5.   
    6.     @NotNull  
    7.     @Length(min = 2, max = 10)  
    8.     private String userName;  
    9.   
    10.     @NotNull  
    11.     @Length(min = 6, max = 20)  
    12.     private String password;  
    13. }  
    • 在方法参数上声明校验注解

    1. @PostMapping("/save")  
    2. public Result saveUser(@RequestBody @Validated UserDTO userDTO) {  
    3.   
    4.     return Result.ok();  
    5. }  

    这种情况下,使用@Valid@Validated都可以

    requestParam/PathVariable参数校验

    GET请求一般会使用requestParam/PathVariable传参。在这种情况下,必须在Controller类上标注@Validated注解,并在入参上声明约束注解 (如@Min等)。如果校验失败,会抛出ConstraintViolationException异常。代码示例如下:

    1. @RequestMapping("/api/user")  
    2. @RestController  
    3. @Validated  
    4. public class UserController {  
    5.   
    6.     @GetMapping("{userId}")  
    7.     public Result detail(@PathVariable("userId") @Min(10000000000000000L) Long userId) {  
    8.   
    9.         UserDTO userDTO = new UserDTO();  
    10.         userDTO.setUserId(userId);   
    11.         userDTO.setUserName("xiaoxiao");   
    12.         return Result.ok(userDTO);  
    13.     }  
    14.   
    15.     @GetMapping("getByAccount")  
    16.     public Result getByAccount(@Length(min = 2, max = 10) @NotNull String  username) {  
    17.   
    18.         UserDTO userDTO = new UserDTO();  
    19.         userDTO.setUserId(10000000000000003L);  
    20.         userDTO.setUserName(username);   
    21.         return Result.ok(userDTO);  
    22.     }  
    23. }  

    统一异常处理

    如果校验失败,会抛出MethodArgumentNotValidException或者ConstraintViolationException异常。在实际项目开发中,通常会用统一异常处理来返回一个更友好的提示。比如我们系统要求无论发送什么异常,http的状态码必须返回200,由业务码去区分系统的异常情况。

    1. @RestControllerAdvice  
    2. public class CommonExceptionHandler {  
    3.   
    4.     @ExceptionHandler({MethodArgumentNotValidException.class})  
    5.     @ResponseStatus(HttpStatus.OK)  
    6.     @ResponseBody  
    7.     public Result handleMethodArgumentNotValidException(MethodArgumentNotValidException ex) {  
    8.         BindingResult bindingResult = ex.getBindingResult();  
    9.         StringBuilder sb = new StringBuilder("校验失败:");  
    10.         for (FieldError fieldError : bindingResult.getFieldErrors()) {  
    11.             sb.append(fieldError.getField()).append(":").append(fieldError.getDefaultMessage()).append(", ");  
    12.         }  
    13.         String msg = sb.toString();  
    14.        return Result.fail(msg);  
    15.     }  
    16.   
    17.     @ExceptionHandler({ConstraintViolationException.class})  
    18.     @ResponseStatus(HttpStatus.OK)  
    19.     @ResponseBody  
    20.     public Result handleConstraintViolationException(ConstraintViolationException ex) {  
    21.         return Result.fail(ex.getMessage());  
    22.     }  
    23. }  

    分组校验 

    在实际项目中,可能多个方法需要使用同一个DTO类来接收参数,而不同方法的校验规则很可能是不一样的。这个时候,简单地在DTO类的字段上加约束注解无法解决这个问题。因此,spring-validation支持了分组校验的功能,专门用来解决这类问题。还是上面的例子,比如保存User的时候,UserId是可空的,但是更新User的时候,UserId的值必须>=10000000000000000L;其它字段的校验规则在两种情况下一样。这个时候使用分组校验的代码示例如下:

    • 约束注解上声明适用的分组信息groups

    1. @Data
    2. public class UserDTO {
    3. // 只限制更新(Update.class)
    4. @Min(value = 10000000000000000L, groups = Update.class)
    5. private Long userId;
    6. // 不为null,长度限制对save、updata都有效
    7. @NotNull(groups = {Save.class, Update.class})
    8. @Length(min = 2, max = 10, groups = {Save.class, Update.class})
    9. private String userName;
    10. @NotNull(groups = {Save.class, Update.class})
    11. @Length(min = 6, max = 20, groups = {Save.class, Update.class})
    12. private String password;
    13. }
    • @Validated注解上指定校验分组

    1. @PostMapping("/save")
    2. public Result saveUser(@RequestBody @Validated(UserDTO.Save.class) UserDTO userDTO) {
    3. return Result.ok();
    4. }
    5. @PostMapping("/update")
    6. public Result updateUser(@RequestBody @Validated(UserDTO.Update.class) UserDTO userDTO) {
    7. return Result.ok();
    8. }

    嵌套校验 

    前面的示例中,DTO类里面的字段都是基本数据类型String类型。但是实际场景中,有可能某个字段也是一个对象,这种情况下,可以使用嵌套校验

    比如,上面保存User信息的时候同时还带有Address信息。需要注意的是,此时DTO类的对应字段必须标记@Valid注解

    1. @Data
    2. public class UserDTO {
    3. @Min(value = 10000000000000000L, groups = Update.class)
    4. private Long userId;
    5. @NotNull(groups = {Save.class, Update.class})
    6. @Length(min = 2, max = 10, groups = {Save.class, Update.class})
    7. private String userName;
    8. @NotNull(groups = {Save.class, Update.class})
    9. @Length(min = 6, max = 20, groups = {Save.class, Update.class})
    10. private String password;
    11. @NotNull(groups = {Save.class, Update.class})
    12. @Valid
    13. private Address addr;
    14. @Data
    15. public static class Address{
    16. @Min(value = 1, groups = Update.class)
    17. private Long addrId;
    18. @NotNull(groups = {Save.class, Update.class})
    19. @Length(min = 2, max = 10, groups = {Save.class, Update.class})
    20. private String addrName;
    21. @NotNull(groups = {Save.class, Update.class})
    22. @Length(min = 2, max = 10, groups = {Save.class, Update.class})
    23. private String location;
    24. }
    25. }

    集合校验 

    如果请求体直接传递了集合给后台,并希望对集合中的每一项都进行参数校验。此时,如果我们直接使用java.util.Collection下的list或者set来接收数据,参数校验并不会生效!我们可以使用自定义list集合来接收参数:

    集合校验会对集合里面的每一项都进行校验,例如List

    字段会对这个list里面的每一个address对象都进行校验。

    • 自定义List集合
    1. public class ValidationList implements List {
    2. @Delegate
    3. @Valid
    4. public List list = new ArrayList<>();
    5. @Override
    6. public String toString() {
    7. return list.toString();
    8. }
    9. }

    @Delegate注解受lombok版本限制,1.18.6以上版本可支持。如果校验不通过,会抛出NotReadablePropertyException,同样可以使用统一异常进行处理。

    一次请求保存多个User对象

    1. @PostMapping("/saveList")
    2. public Result saveList(@RequestBody @Validated(UserDTO.Save.class) ValidationList userList) {
    3. return Result.ok();
    4. }

    @Valid和@Validated区别

     

     

  • 相关阅读:
    继GitHub的Copilot收费后,亚马逊推出了 CodeWhisperer,感觉不错哟!
    1.ts介绍
    C语言进阶第三课-----------指针的进阶----------后续版
    先验 & 后验 & 似然估计
    【亲测】阿里云ECS搭建《幻兽帕鲁》联机服务器
    Day41信息收集
    vscode的 c++ 环境搭建
    9月8日作业
    STM32CUBEIDE编译库函数项目及库文件调用
    牛血清白蛋白/人血清白蛋白/卵清白蛋白/小鼠血清白蛋白纳米粒改性Angiopep血管肽(2022已更新)
  • 原文地址:https://blog.csdn.net/qq_48078182/article/details/126180215