后端数据校验与全局异常处理

三 JSR303规范 后端数据校验与全局异常处理

  1. 添加校验注解:javax.validation.constraints包里的校验注解

@Data
@TableName("pms_brand")
public class BrandEntity implements Serializable {
     
	private static final long serialVersionUID = 1L;

	/**
	 * 品牌id
	 */
	@NotNull(message = "修改必须指定品牌id")
	@Null(message = "新增不能指定id")
	@TableId
	private Long brandId;
	/**
	 * 品牌名
	 */
	@NotBlank(message = "品牌名必须提交")
	private String name;
	/**
	 * 品牌logo地址
	 */
	@NotBlank
	@URL(message = "logo必须是一个合法的url地址")
	private String logo;
	/**
	 * 介绍
	 */
	private String descript;
	/**
	 * 显示状态[0-不显示;1-显示]
	 */
//	@Pattern()
	@NotNull
  	@ListValue(vals={
     0,1})
	private Integer showStatus;
	/**
	 * 检索首字母
	 */
	@NotEmpty
	@Pattern(regexp="^[a-zA-Z]$",message = "检索首字母必须是一个字母")
	private String firstLetter;
	/**
	 * 排序
	 */
	@NotNull
	@Min(value = 0,message = "排序必须大于等于0")
	private Integer sort;
}

  1. 开启校验功能 @Valid
@RequestMapping("/save")
//@RequiresPermissions("product:brand:save")
public R save(@Valid @RequestBody BrandEntity brand,BindingResult result){
     
    if(result.hasErrors()){
     
        Map<String,String> map = new HashMap<>();
        //1、获取校验的错误结果
        result.getFieldErrors().forEach((item)->{
     
            //FieldError 获取到错误提示
            String message = item.getDefaultMessage();
            //获取错误的属性的名字
            String field = item.getField();
            map.put(field,message);
        });
        return R.error(400,"提交的数据不合法").put("data",map);
    }else {
     
    }
    brandService.save(brand);
    return R.ok();
}
  1. 全局异常处理类
/**
 * 集中处理所有异常
 */
@Slf4j
//@ResponseBody
//@ControllerAdvice(basePackages = "com.atguigu.gulimall.product.controller")
@RestControllerAdvice(basePackages = "com.atguigu.gulimall.product.controller")
public class GulimallExceptionControllerAdvice {
     
    @ExceptionHandler(value= MethodArgumentNotValidException.class)
    public R handleVaildException(MethodArgumentNotValidException e){
     
        log.error("数据校验出现问题{},异常类型:{}",e.getMessage(),e.getClass());
        BindingResult bindingResult = e.getBindingResult();
        Map<String,String> errorMap = new HashMap<>();
        bindingResult.getFieldErrors().forEach((fieldError)->{
     
            errorMap.put(fieldError.getField(),fieldError.getDefaultMessage());
        });
        return R.error(BizCodeEnume.VAILD_EXCEPTION.getCode(),BizCodeEnume.VAILD_EXCEPTION.getMsg()).put("data",errorMap);
    }
    @ExceptionHandler(value = Throwable.class)
    public R handleException(Throwable throwable){
     
        log.error("错误:",throwable);
        return R.error(BizCodeEnume.UNKNOW_EXCEPTION.getCode(),BizCodeEnume.UNKNOW_EXCEPTION.getMsg());
    }
}
  1. 异常code与message枚举类定义
public enum BizCodeEnume {
     
    UNKNOW_EXCEPTION(10000,"系统未知异常"),
    VAILD_EXCEPTION(10001,"参数格式校验失败");

    private int code;
    private String msg;
    BizCodeEnume(int code,String msg){
     
        this.code = code;
        this.msg = msg;
    }

    public int getCode() {
     
        return code;
    }

    public String getMsg() {
     
        return msg;
    }
}
  1. 分组校验 groups
    1. 校验注解编辑gorups属性
    2. 使用@validated(AddGroup.class),选择操作组的类型

分组的接口

AddGroup.java
public interface AddGroup {
     
}

UpdateGroup.java
public interface UpdateGroup {
     
}

实体类

public class BrandEntity implements Serializable {
     
	private static final long serialVersionUID = 1L;

	/**
	 * 品牌id
	 */
	@NotNull(message = "修改必须指定品牌id",groups = {
     UpdateGroup.class})
	@Null(message = "新增不能指定id",groups = {
     AddGroup.class})
	@TableId
	private Long brandId;
	/**
	 * 品牌名
	 */
	@NotBlank(message = "品牌名必须提交",groups = {
     AddGroup.class,UpdateGroup.class})
	private String name;
	/**
	 * 品牌logo地址
	 */
	@NotBlank(groups = {
     AddGroup.class})
	@URL(message = "logo必须是一个合法的url地址",groups={
     AddGroup.class,UpdateGroup.class})
	private String logo;
	/**
	 * 介绍
	 */
	private String descript;
	/**
	 * 显示状态[0-不显示;1-显示]
	 */
//	@Pattern()
	@NotNull(groups = {
     AddGroup.class, UpdateStatusGroup.class})
  	@ListValue(vals={
     0,1},groups = {
     AddGroup.class, UpdateStatusGroup.class})
	private Integer showStatus;
	/**
	 * 检索首字母
	 */
	@NotEmpty(groups={
     AddGroup.class})
	@Pattern(regexp="^[a-zA-Z]$",message = "检索首字母必须是一个字母",groups={
     AddGroup.class,UpdateGroup.class})
	private String firstLetter;
	/**
	 * 排序
	 */
	@NotNull(groups={
     AddGroup.class})
	@Min(value = 0,message = "排序必须大于等于0",groups={
     AddGroup.class,UpdateGroup.class})
	private Integer sort;
}

使用validated注解和分组,只有存在AddGroup中校验注解生效
6. 自定义校验
1. 创建自定义校验注解,满足JSR303规范
2. 编写自定义校验器
3. 关联自定义校验器和自定义校验注解

自定义注解

@Documented //文档注解
@Constraint(validatedBy = {
      ListValueConstraintValidator.class }) //绑定校验器[可以指定多个不同的校验器,适配不同类型的校验]
@Target({
      METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER, TYPE_USE }) //注解作用目标
@Retention(RUNTIME) //注解作用周期
public @interface ListValue {
     
    //提示信息
    String message() default "{****}";
    //分组
    Class<?>[] groups() default {
      };
    
    Class<? extends Payload>[] payload() default {
      };
    默认值
    int[] vals() default {
      };
}

ValidationMessages.properties 自定义配置文件

****=默认提示信息

自定义校验器

ListValueConstraintValidator implements ConstraintValidator<ListValue,Integer> {
     

    private Set<Integer> set = new HashSet<>();
    //初始化方法
    @Override
    public void initialize(ListValue constraintAnnotation) {
     

        int[] vals = constraintAnnotation.vals();
        for (int val : vals) {
     
            set.add(val);
        }

    }

    //判断是否校验成功

    /**
     *
     * @param value 需要校验的值
     * @param context
     * @return
     */
    @Override
    public boolean isValid(Integer value, ConstraintValidatorContext context) {
     

        return set.contains(value);
    }
}

你可能感兴趣的:(java,spring)