所有分类
  • 所有分类
  • 未分类

SpringBoot-用hibernate validator手动校验

简介

说明

本文介绍如何使用validator进行手动校验。

validator可以进行自动校验,也可以进行手动校验,结果是一模一样的。自动校验见下方“自动校验”。

自动校验

自动校验的方法:项目中我们经常在请求类上加@NotNull,@NotBlank之类的注解,结合@Valid或者@Validated,即可自动对字段进行校验。自动校验的用法见:SpringBoot-@Valid与@Validated的区别 – 自学精灵

效果展示1:基础功能

代码

Controller

package com.knife.example.business.user.controller;

import com.knife.example.business.user.bo.UserAddBO;
import com.knife.example.business.user.dto.UserDTO;
import com.knife.example.business.user.vo.UserVO;
import com.knife.example.common.group.ValidGroup;
import com.knife.example.common.util.ValidateUtil;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import org.springframework.beans.BeanUtils;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.time.LocalDateTime;
 
@Api(tags = "手动校验")
@RequestMapping("manualValidate")
@RestController
public class ManualController {
    @ApiOperation("添加")
    @PostMapping("/add")
    public UserVO add(@RequestBody UserAddBO userAddBO) {
        UserDTO userDTO = new UserDTO();
        BeanUtils.copyProperties(userAddBO, userDTO);
        // ValidUtil.validate(userDTO);
        ValidateUtil.validate(userDTO, ValidGroup.Add.class);
 
        // 将数据写到数据库
        userDTO.setId(1L);
        userDTO.setCreateTime(LocalDateTime.now());
        userDTO.setUpdateTime(LocalDateTime.now());
 
        UserVO userVO = new UserVO();
        BeanUtils.copyProperties(userDTO, userVO);
        return userVO;
    }
}

BO

package com.knife.example.business.user.bo;
 
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
 
@Data
@ApiModel("添加用户")
public class UserAddBO {
    @ApiModelProperty(value = "用户名", required = true)
    private String userName;
 
    @ApiModelProperty("昵称")
    private String nickName;
 
    @ApiModelProperty("邮箱")
    private String email;
}

DTO

package com.knife.example.business.user.dto;
 
import com.knife.example.common.group.ValidGroup;
import lombok.Data;
 
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Null;
import java.time.LocalDateTime;
 
@Data
public class UserDTO {
    /**
     * 用户id
     */
    @Null(groups = ValidGroup.Add.class, message = "用户id必须为空")
    @NotNull(groups = ValidGroup.Edit.class, message = "用户id不能为空")
    private Long id;
 
    /**
     * 用户名
     */
    @NotBlank(groups = {ValidGroup.Add.class, ValidGroup.Edit.class},
            message = "用户名不能为空")
    private String userName;
 
    /**
     * 昵称
     */
    private String nickName;
 
    /**
     * 邮箱
     */
    @NotBlank(message = "邮箱不能为空")
    private String email;
 
    /**
     * 创建时间
     */
    @Null(message = "创建时间必须为空")
    private LocalDateTime createTime;
 
    /**
     * 修改时间
     */
    @NotNull(message = "更新时间不能为空")
    private LocalDateTime updateTime;
}

测试

测试添加接口缺少字段的场景

访问:

请求体为:

{
  "email": "",
  "nickName": "",
  "userName": ""
}

效果展示2:数据嵌套

代码

Controller

package com.knife.example.business.nested.controller;

import com.knife.example.business.nested.bo.UserBO;
import com.knife.example.business.nested.group.IGroupA;
import com.knife.example.business.nested.group.IGroupAll;
import com.knife.example.business.nested.group.IGroupB;
import com.knife.example.common.util.ValidateUtil;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@Api(tags = "手动校验嵌套")
@RestController
@RequestMapping("nested")
public class ValidatedWithGroupController {
    @ApiOperation("不使用组")
    @PostMapping("withoutGroup")
    public UserBO withoutGroup(@RequestBody UserBO userBO) {
        ValidateUtil.validate(userBO);
        return userBO;
    }

    @ApiOperation("使用组:GroupA")
    @PostMapping("groupA")
    public UserBO groupA(@RequestBody UserBO userBO) {
        ValidateUtil.validate(userBO, IGroupA.class);
        return userBO;
    }
 
    @ApiOperation("使用组:GroupB")
    @PostMapping("groupB")
    public UserBO groupB(@RequestBody UserBO userBO) {
        ValidateUtil.validate(userBO, IGroupB.class);
        return userBO;
    }
 
    @ApiOperation("使用组:GroupAll")
    @PostMapping("groupAll")
    public UserBO groupAll(@Validated({IGroupAll.class}) @RequestBody UserBO userBO) {
        ValidateUtil.validate(userBO, IGroupAll.class);
        return userBO;
    }
}

BO

package com.knife.example.business.nested.bo;

import com.knife.example.business.nested.group.IGroupA;
import com.knife.example.business.nested.group.IGroupB;
import lombok.Data;

import javax.validation.Valid;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.NotNull;
import java.util.List;
 
@Data
public class UserBO {
    @NotBlank(message = "名字不能为空")
    private String name;

    @NotNull(message = "年龄不能为空", groups = {IGroupA.class})
    private Integer age;

    @NotEmpty(message = "密码不能为空", groups = {IGroupB.class})
    private String password;

    @NotEmpty(message = "分数不能为空", groups = {IGroupA.class, IGroupB.class})
    private List<Integer> scoreArray;

    @Valid
    @NotNull(message = "账户不能为空")
    private AccountBO accountBO;

    @Valid
    @NotEmpty(message = "账户列表不能为空")
    private List<AccountBO> accountBOList;
}
package com.knife.example.business.nested.bo;
 
import com.knife.example.business.nested.group.IGroupA;
import com.knife.example.business.nested.group.IGroupB;
import lombok.Data;

import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.NotNull;

@Data
public class AccountBO {
    @NotNull(message = "账户ID不能为空", groups = {IGroupA.class} )
    private Long id;

    @NotBlank(message = "电话号码不能为空")
    private String phoneNumber;

    private String[] emails;
}

测试

测试1:缺参数测试不分组

可以发现:校验生效了(列表形式的包装类字段也生效了)。这样不加Group时,只校验没有Group的字段。(跟自动校验的结果一样)。

请求体:

{
  "accountBO": {
    "emails": [],
    "id": 2,
    "phoneNumber": ""
  },
  "accountBOList": [
    {
      "emails": [],
      "id": null,
      "phoneNumber": ""
    }
  ],
  "age": 0,
  "name": "",
  "password": "",
  "scoreArray": []
}

测试2:缺字段测试GroupA

可以发现:GroupA的校验生效了(列表形式的包装类字段也生效了)。跟使用注解自动校验的结果一模一样,见:这里

请求体:

{
  "accountBO": {
    "emails": [],
    "id": 2,
    "phoneNumber": ""
  },
  "accountBOList": [
    {
      "emails": [],
      "id": null,
      "phoneNumber": ""
    }
  ],
  "age": 0,
  "name": "",
  "password": "",
  "scoreArray": []
}

测试3:缺字段测试GroupAll

可以发现:可以看到:走的校验逻辑是除了IGroupA和IGroupB注解的字段的逻辑。

核心代码

此内容仅限VIP查看,请先
0

评论0

请先

显示验证码
没有账号?注册  忘记密码?

社交账号快速登录