springboot自带的@Validated注解的使用-笔记

springboot自带的@Validated注解的使用-笔记

@Validated介绍

@Validated 是spring提供的,提供了分组校验功能,只能用在类型、方法、参数上;
如果类名上和参数上都加了这个注解,以参数上为准;

@Validated全局校验抛的异常是 ConstraintViolationException
@Validated分组校验抛的异常是 MethodArgumentNotValidException;

@Validated全局校验测试

@Validated
@RestController
public class Test05292Api {

    @PostMapping("/t1234")
    public UserReq test1234(@Valid @RequestBody UserReq req, BindingResult bindingResult) {
        log.info("req:{}", req.toString());
        return req;
    }
    
    /**
     * @Validated全局校验
     * 对list参数校验,返回的异常信息如下
     * addUsers.list[0].phone: 必须是手机
     * addUsers.list[0].company.id: 不能为null
     * addUsers.list[0].testCount: 需要在1和99之间
     */
    @PostMapping("/users")
    public List<UserReq> addUsers(@Valid @RequestBody List<UserReq> list) {
        log.info("list:{}", list.toString());

        return list;
    }
}

@Data
public class UserReq {

    @NotBlank
    @Size(max = 30,min = 2,message = "长度需要在2-30之间")
    private String account;

    @Email
    private String email;

    @Pattern(regexp = "(?:0|86|\\+86)?1[3-9]\\d{9}",message = "必须是手机号")
    private String phone;

    /**
     * 必须是一个过去的或当前的日期
     */
    @PastOrPresent
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime  startDt;

    /**
     * 必须是一个将来的或当前的日期
     */
    @FutureOrPresent
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime endDt;

    /**
     * 嵌套校验
     */
    @Valid
    private CompanyDTO company;

    /**
     * hibernate.validator包提供的 范围限制注解 @Range ,限制数值的大小
     */
    @Range(min = 1,max = 99)
    private Integer testCount;

}

@Data
public class CompanyDTO {
    @NotNull
    private Integer id;

    @NotBlank
    @Length(max = 30,min=4,message = "长度需要在4-30之间")
    private String name;

    @Pattern(regexp ="(010|02\\d|0[3-9]\\d{2})-?(\\d{6,8})",message = "必须是国内固定电话号码")
    private String tel;
}
{
    "code": 9996,
    "msg": "参数校验异常",
    "content": "addUsers.list[0].testCount: 需要在1和99之间, addUsers.list[0].company.id: 不能为null",
    "timestamp": 1685349934258,
    "traceDateTime": "2023-05-29 16:45:34"
}

需要用ExceptionHandler手动处理ConstraintViolationException

/**
     * Validated全局自动校验,要@Validated加在Controller类名上,同时手动使用BindingResult
     */
    @ExceptionHandler({ConstraintViolationException.class})
    public ResultVO constraintViolationExceptionHandler(ConstraintViolationException e) {
        log.error("Validated全局自动校验出的异常",e);
        //校验框架会自动返回多条约束提示信息的拼接
        //格式如 test1234.req.company.name: 长度需要在4-30之间,test1234.req.account: 长度需要在2-30之间
        //(逗号拼接的字符串  接口方法名.参数名.子参数对象.子参数属性:描述)
        return new ResultVO<>(ResultCodeEnum.VALIDATE_FAILED,
                e.getMessage());
}

总结

@Validated加在类名上,且api参数里有BindingResult时,
 会抛出javax.validation.ConstraintViolationException ,
 异常信息返回如下
  test1234.req.company.name: 长度需要在4-30之间,
 test1234.req.account: 长度需要在2-30之间
 格式的错误信息 (逗号拼接的字符串 , 接口方法名.参数名.子参数对象.子参数属性) ;
 
 @Validated全局校验 可以和 @Valid嵌套校验一起使用,
 即 @Validated全局校验时,也能处理复合JavaBean的子对象的子属性的参数校验,
 需要JavaBean里的子对象上加@Valid注解

@Validated分组校验测试

/**
 * @Validated({UpdateActionGroup.class})分组校验测试,抛的异常是 MethodArgumentNotValidException
 */
@RestController
@Slf4j
public class Test05293Api {

    /**
     * 新增时不校验id
     * @param req UserTestReq1
     * @return UserTestResp
     *
     * 触发校验时返回信息为
     *  userTestReq1.name不能为空 或
     *  userTestReq1.age不能为null
     */
    @PostMapping("/addUser")
    public UserTestResp  addUser(@RequestBody @Validated
                                             UserTestReq1 req){
        log.info("req:{}",req);
        return UserTestResp.builder()
                .age(req.getAge())
                .name(req.getName())
                .build();
    }
    
    /**
     * 更新时校验id
     * @param req UserTestReq1   在 UpdateActionGroup下只校验id字段
     * @return UserTestResp
     * {
     *     "code": 9996,
     *     "msg": "参数校验异常",
     *     "content": "userTestReq1.id不能为null",
     *     "timestamp": 1685344365874,
     *     "traceDateTime": "2023-05-29 15:12:45"
     * }
     */
    @PostMapping("/alterUser")
    public UserTestResp  alterUser(@RequestBody  @Validated({UpdateActionGroup.class})
                                               UserTestReq1 req ){
        log.info("req:{}",req);
        return UserTestResp.builder()
                .id(req.getId())
                .age(req.getAge())
                .name(req.getName())
                .build();
    }
}    
import lombok.Data;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotNull;

/**
 * 分组校验 和 嵌套校验 不能同时使用,否则会使得分组校验失效...
 */
@Data
public class UserTestReq1 {
    /**
     *  在 UpdateActionGroup下只校验id字段
     */
    @NotNull(groups = UpdateActionGroup.class)
    private Long id;
    
    /**
     * 在默认情况下只校验 name字段和age字段
     */
    @NotBlank
    private String name;
    @NotNull
    private Integer age;
    
    //............这里可以增加更多字段,然后对字段设置分组
    
    //分组校验可以高度复用 RequestJavaBean 。
    //在做一些固定对象数据的增删改查时,用到的都是同一套参数,
    //只是每次校验的分组不同,可以使用 @Validated({xxxxGroup.class})分组校验
}

/**
 * 创建1个分组 UpdateActionGroup
 */
public interface UpdateActionGroup {
}

需要用ExceptionHandler手动处理MethodArgumentNotValidException

@ExceptionHandler({MethodArgumentNotValidException.class})
    public ResultVO<String> MethodArgumentNotValidExceptionHandler(MethodArgumentNotValidException e) {
        log.error("自动抛的MethodArgumentNotValidException",e);
        // 从异常对象中拿到ObjectError对象
        ObjectError objectError = e.getBindingResult().getAllErrors().get(0);
        log.error("objectError:{}", JSONObject.toJSONString(objectError, true));
        String field = String.valueOf(JSONPath.eval(objectError, "$.field"));
        log.error("field:{}", field);
        String objName = objectError.getObjectName();
        System.err.println(objName);
        // 然后提取错误提示信息进行返回
        return new ResultVO<>(ResultCodeEnum.VALIDATE_FAILED,
                objName + "." + field + objectError.getDefaultMessage());
    }
{
    "code": 9996,
    "msg": "参数校验异常",
    "content": "userTestReq1.id不能为null",
    "timestamp": 1685349830928,
    "traceDateTime": "2023-05-29 16:43:50"
}

总结

@Validated分组校验可以高度复用 RequestJavaBean;
但是不能和 @Vaild嵌套校验功能一起使用;

@Validated分组校验更适合对同一类数据的一套参数,进行分组的定制校验;
但是如果分组校验的参数bean的字段越来越多时,如 UserTestReq1,
可能需要按需创建更多的分组接口,长期下去对类代码的可读性会变差,要使用的话,最好都加上明确的注释,以便于维护代码。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
`@Validated`注解Spring框架提供的一种参数校验注解。它可以用来替代Java标准库中的`@Valid`注解,并且支持分组校验等功能。 `@Validated`注解可以用在类、方法、构造函数、方法参数以及方法返回值上。它的作用是启用方法参数校验。当方法被调用时,如果参数上标注了`@Validated`注解,则对该参数进行校验。如果校验失败,则抛出`MethodArgumentNotValidException`异常。 除了支持Java标准库中的参数校验注解之外,`@Validated`注解还支持分组校验功能。即可以根据不同的校验场景,对同一个类中的不同属性进行不同的校验。例如: ```java public interface Update { } public interface Create { } public class User { @NotNull(groups = Update.class) private Long id; @NotBlank(groups = {Create.class, Update.class}) private String name; @Min(value = 18, message = "年龄不能小于18岁", groups = {Create.class, Update.class}) private Integer age; // 省略getter和setter方法 } ``` 在上面的代码中,定义了两个分组`Create`和`Update`,并在`User`类中使用了分组校验注解。其中,`id`属性只在更新用户信息时进行校验,`name`和`age`属性在创建和更新用户信息时都需要进行校验。当进行校验时,需要指定对应的分组,例如: ```java @RestController @RequestMapping("/user") @Validated public class UserController { @PostMapping("/create") public ResponseEntity createUser(@RequestBody @Validated(Create.class) User user) { // ... } @PostMapping("/update") public ResponseEntity updateUser(@RequestBody @Validated(Update.class) User user) { // ... } } ``` 在上面的代码中,`@Validated(Create.class)`注解表示对`User`对象中标注了`Create`分组的属性进行校验,`@Validated(Update.class)`注解表示对`User`对象中标注了`Update`分组的属性进行校验。 综上所述,`@Validated`注解可以用来替代Java标准库中的`@Valid`注解,并且支持分组校验等功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ThinkPet

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值