最近发现前端传来的参数,用if来判断太繁琐了,改用注解形式,这里介绍一下如何自定义校验注解,前面我有篇文章介绍了javax.validation工具包的简单用法,不懂的可以看看。
pom引入
javax.validation
validation-api
2.0.1.Final
compile
true
org.hibernate.validator
hibernate-validator
实体类对象
package com.wrc.gulimall.gulimallproduct.entity;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import java.io.Serializable;
import com.wrc.common.valid.AddGroup;
import com.wrc.common.valid.ListValue;
import com.wrc.common.valid.UpdateGroup;
import com.wrc.common.valid.UpdateStatusGroup;
import lombok.Data;
import org.hibernate.validator.constraints.URL;
import javax.validation.constraints.*;
/**
* 品牌
*
* @author wrc
* @email
* @date 2019-10-01 21:08:49
*/
@Data
@TableName("pms_brand")
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-显示]
*/
@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;
}
@NotNull(message = "修改必须指定品牌id",groups = {UpdateGroup.class}
groups是分组的意思,我们在contreller层方法上表明这个注解生效于哪个方法,通过这个分组来管理对象的属性对哪个起效
分组的指定的类,addGroup,UpdateGroup 需要自己创建。
创建空的接口就行,不需要实现。
现在我们先测试一下分组的功能
测试修改这个功能
由于我们调用的是修改方法,上面实体类使用了
@NotNull(message = "修改必须指定品牌id",groups = {UpdateGroup.class})这个注解
加上id 校验成功。
但是如果想要校验别的格式的数据,可以自定义注解。
@ListValue注解的意思是校验数据是否是0或者1,生效分组是新增和修改显示状态
package com.wrc.common.valid;
import javax.validation.Constraint;
import javax.validation.Payload;
import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import static java.lang.annotation.ElementType.*;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
@Documented
//指定注解的实现类
@Constraint(validatedBy = { ListValueConstraintValidator.class })
@Target({ METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER, TYPE_USE })
@Retention(RUNTIME)
//自定义 注解
public @interface ListValue {
//message 指定如果校验不合格从这里读取返回数据
String message() default "{com.wrc.common.valid.ListValue.message}";
//这剩下3个方法 例行要偶 指定分组 。。。。。 注解校验数据
Class>[] groups() default { };
Class extends Payload>[] payload() default { };
int[] vals() default { };
}
package com.wrc.common.valid;
import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;
import java.util.HashSet;
import java.util.Set;
public class ListValueConstraintValidator implements ConstraintValidator {
private Set 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);
}
}
com.wrc.common.valid.ListValue.message 文件
测试一下
枚举类
package com.wrc.common.exception;
/***
* 错误码和错误信息定义类
* 1. 错误码定义规则为5为数字
* 2. 前两位表示业务场景,最后三位表示错误码。例如:100001。10:通用 001:系统未知异常
* 3. 维护错误码后需要维护错误描述,将他们定义为枚举形式
* 错误码列表:
* 10: 通用
* 001:参数格式校验
* 11: 商品
* 12: 订单
* 13: 购物车
* 14: 物流
*
*
*/
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;
}
}
package com.wrc.gulimall.gulimallproduct.exception;
import com.wrc.common.exception.BizCodeEnume;
import com.wrc.common.utils.R;
import lombok.extern.slf4j.Slf4j;
import org.springframework.validation.BindingResult;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import java.util.HashMap;
import java.util.Map;
/**
* @Author: wrc
* @Classname ExceptionControllerAdvice
* @Description TODO
* @Date 2020/7/24 16:18
* @Created wrc
*/
@Slf4j
指定接收哪个包下的异常
@RestControllerAdvice(basePackages = "com.wrc.gulimall.gulimallproduct.controller")
public class ExceptionControllerAdvice {
/** 处理product服务 非空异常
* @ExceptionHandler 指定接收哪个异常
* @return
*/
@ExceptionHandler(value = MethodArgumentNotValidException.class)
public R handleValidException(MethodArgumentNotValidException e) {
BindingResult bindingResult = e.getBindingResult();
Map map = new HashMap<>();
if(bindingResult.hasErrors()){
//1、获取校验的错误结果
bindingResult.getFieldErrors().forEach((item)->{
//FieldError 获取到错误提示
String message = item.getDefaultMessage();
//获取错误的属性的名字
String field = item.getField();
map.put(field,message);
});
}
return R.error(BizCodeEnume.VAILD_EXCEPTION.getCode(),BizCodeEnume.VAILD_EXCEPTION.getMsg()).put("data",map);
}
@ExceptionHandler(value = Throwable.class)
public R handleException(MethodArgumentNotValidException e) {
return R.error(BizCodeEnume.UNKNOW_EXCEPTION.getCode(),BizCodeEnume.UNKNOW_EXCEPTION.getMsg());
}
}