zl程序教程

您现在的位置是:首页 >  工具

当前栏目

@Valid、@Validated 、正则验证工具

工具 验证 正则 valid
2023-09-27 14:19:41 时间
  • 使用方法

 

  • 源码跟踪:
springframework.web--》WebDataBinder-》validate
springframework validation---RequestResponseBodyMethodProcessor--》resolveArgument--》validateIfApplicable
hibernate validator---ConstraintHelper
  • 自己实现工具类
 1 import com.amp.base.exception.BusinessException;
 2 import com.google.common.base.Joiner;
 3 
 4 import javax.validation.ConstraintViolation;
 5 import javax.validation.Validation;
 6 import javax.validation.Validator;
 7 import java.util.*;
 8 
 9 /**
10  * 校验工具类
11  */
12 public class ValidatorUtil {
13     private static Validator validator = Validation.buildDefaultValidatorFactory().getValidator();
14 
15     /**
16      * hibernate validator校验
17      *
18      * @param obj 校验对象
19      * @return void
20      */
21     public static <T> void validate(T obj) {
22         Set<ConstraintViolation<T>> constraintViolations = validator.validate(obj);
23         List<String> list = new ArrayList<>();
24         // 抛出检验异常
25         if (constraintViolations.size() > 0) {
26             for (ConstraintViolation constraintViolation : constraintViolations) {
27                 list.add(constraintViolation.getMessage());
28             }
29         }
30         if (list != null && list.size() > 0) {
31             throw new BusinessException(Joiner.on(",").skipNulls().join(list));
32         }
33     }
34 
35     /**
36      * 校验bean返回map形式的校验结果
37      *
38      * @param obj bean
39      * @return java.util.Map<java.lang.String , java.lang.String>
40      */
41     public static <T> Map<String, String> validateToMap(T obj) {
42         Set<ConstraintViolation<T>> constraintViolations = validator.validate(obj);
43         Map<String, String> map = new HashMap<>(constraintViolations.size());
44         if (constraintViolations.size() > 0) {
45             for (ConstraintViolation constraintViolation : constraintViolations) {
46                 map.put(constraintViolation.getPropertyPath().toString(), constraintViolation.getMessage());
47             }
48         }
49         return map;
50     }
51 
52     /**
53      * 校验bean返回map形式的校验结果
54      *
55      * @param obj bean
56      * @return java.util.Map<java.lang.String , java.lang.String>
57      */
58     public static <T> List<String> validateToList(T obj) {
59         Set<ConstraintViolation<T>> constraintViolations = validator.validate(obj);
60         List<String> list = new ArrayList<>();
61         if (constraintViolations.size() > 0) {
62             for (ConstraintViolation constraintViolation : constraintViolations) {
63                 list.add(constraintViolation.getMessage());
64             }
65         }
66         return list;
67     }
68     /**
69      * 校验bean返回String形式的校验结果
70      *
71      * @param obj bean
72      * @return java.util.Map<java.lang.String , java.lang.String>
73      */
74     public static <T> String validateToString(T obj) {
75         Set<ConstraintViolation<T>> constraintViolations = validator.validate(obj);
76         List<String> list = new ArrayList<>();
77         if (constraintViolations.size() > 0) {
78             for (ConstraintViolation constraintViolation : constraintViolations) {
79                 list.add(constraintViolation.getMessage());
80             }
81         }
82         return Joiner.on(",").skipNulls().join(list);
83     }
84 }
View Code
  • 自己实现正则验证工具类
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * 正则验证工具类
 *
 */
public final class ValidateUtil {

    /** 整数 */
    public static final String V_INTEGER = "^-?[1-9]\\d*$";

    /** 正整数 */
    public static final String V_Z_INDEX = "^[1-9]\\d*$";

    /** 负整数 */
    public static final String V_NEGATIVE_INTEGER = "^-[1-9]\\d*$";

    /** 中英文和数字*/
    public static final String V_ZY_NUM = "^([\u4E00-\uFA29]|[\uE7C7-\uE7F3]|[a-zA-Z0-9_]){1,50}$";

    /** 中英文*/
    public static final String V_ZY = "^[a-zA-Z\\u4e00-\\u9fa5]+$";
    /** 数字 */
    public static final String V_NUMBER = "^([+-]?)\\d*\\.?\\d+$";

    /** 正数 */
    public static final String V_POSITIVE_NUMBER = "^[1-9]\\d*|0$";

    /** 负数 */
    public static final String V_NEGATINE_NUMBER = "^-[1-9]\\d*|0$";

    /** 浮点数 */
    public static final String V_FLOAT = "^([+-]?)\\d*\\.\\d+$";

    /** 正浮点数 */
    public static final String V_POSTTIVE_FLOAT = "^[1-9]\\d*.\\d*|0.\\d*[1-9]\\d*$";

    /** 负浮点数 */
    public static final String V_NEGATIVE_FLOAT = "^-([1-9]\\d*.\\d*|0.\\d*[1-9]\\d*)$";

    /** 非负浮点数(正浮点数 + 0) */
    public static final String V_UNPOSITIVE_FLOAT = "^[1-9]\\d*.\\d*|0.\\d*[1-9]\\d*|0?.0+|0$";

    /** 非正浮点数(负浮点数 + 0) */
    public static final String V_UN_NEGATIVE_FLOAT = "^(-([1-9]\\d*.\\d*|0.\\d*[1-9]\\d*))|0?.0+|0$";

    /** 邮件 */
    public static final String V_EMAIL = "^\\w+((-\\w+)|(\\.\\w+))*\\@[A-Za-z0-9]+((\\.|-)[A-Za-z0-9]+)*\\.[A-Za-z0-9]+$";

    /** 颜色 */
    public static final String V_COLOR = "^[a-fA-F0-9]{6}$";

    /** url */
    public static final String V_URL = "^http[s]?:\\/\\/([\\w-]+\\.)+[\\w-]+([\\w-./?%&=]*)?$";

    /** 仅中文 */
    public static final String V_CHINESE = "^[\\u4E00-\\u9FA5\\uF900-\\uFA2D]+$";

    /** 仅ACSII字符 */
    public static final String V_ASCII = "^[\\x00-\\xFF]+$";

    /** 邮编 */
    public static final String V_ZIPCODE = "^\\d{6}$";

    /** 手机 */
    public static final String V_PHONE = "^1[3456789]\\d{9}$";

    /** 电话 */
    public static final String V_MOBILE = "^(\\(\\d{3,4}\\)|\\d{3,4}-|\\s)?\\d{7,14}$";

    /** ip地址 */
    public static final String V_IP4 = "^(25[0-5]|2[0-4]\\d|[0-1]\\d{2}|[1-9]?\\d)\\.(25[0-5]|2[0-4]\\d|[0-1]\\d{2}|[1-9]?\\d)\\.(25[0-5]|2[0-4]\\d|[0-1]\\d{2}|[1-9]?\\d)\\.(25[0-5]|2[0-4]\\d|[0-1]\\d{2}|[1-9]?\\d)$";

    /** 非空 */
    public static final String V_NOTEMPTY = "^\\S+$";

    /** 图片 */
    public static final String V_PICTURE = "(.*)\\.(jpg|bmp|gif|ico|pcx|jpeg|tif|png|raw|tga)$";

    /** 压缩文件 */
    public static final String V_RAR = "(.*)\\.(rar|zip|7zip|tgz)$";

    /** 年月日期 */
    public static final String V_YEAR_MONTH = "^(([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3})-(0[1234567879]|1[012]))$";
    /** QQ号码 */
    public static final String V_QQ_NUMBER = "^[1-9]*[1-9][0-9]*$";

    /** 电话号码的函数(包括验证国内区号,国际区号,分机号) */
    public static final String V_TEL = "^(([0\\+]\\d{2,3}-)?(0\\d{2,3})-)?(\\d{7,8})(-(\\d{3,}))?$";

    /** 用来用户注册。匹配由数字、26个英文字母或者下划线组成的字符串 */
    public static final String V_USERNAME = "^\\w+$";

    /** 字母 */
    public static final String V_LETTER = "^[A-Za-z]+$";

    /** 大写字母 */
    public static final String V_LETTER_U = "^[A-Z]+$";

    /** 小写字母 */
    public static final String V_LETTER_I = "^[a-z]+$";

    /** 身份证 */
    public static final String V_IDCARD = "^(\\d{15}$|^\\d{18}$|^\\d{17}(\\d|X|x))$";

    /** 验证密码(数字和英文同时存在) */
    public static final String V_P_REG = "[A-Za-z]+[0-9]";

    /** 验证密码长度(6-18位) */
    public static final String V_P_LENGTH = "^\\d{6,18}$";

    /** 验证一个月的31天 */
    public static final String V_31DAYS = "^((0?[1-9])|((1|2)[0-9])|30|31)$";

    /**
     * 验证价格
     */
    public static final String V_PRICE = "^([1-9]{1}[0-9]{0,}(\\.[0-9]{0,2})?|0(\\.[0-9]{0,2})?|\\.[0-9]{1,2})$";

    /**
     * 验证金额
     */
    public static final String V_MONEY = "^(([1-9][0-9]*)|(([0]\\.\\d{1,2}|[1-9][0-9]*\\.\\d{1,2})))$";
    /**
     * 中国人名
     */
    public static final String V_PERSON_NAME = "[\\u4E00-\\u9FA5]{2,5}(?:·[\\u4E00-\\u9FA5]{2,5})*";

    /**
     * 验证是否以数学运算符开始
     */
    public static final String V_MATH_OPERATOR = "^[+ / * -](.*?)";

    /**
     * 是否是正数,或者小数不超过两位小数点
     */
    public static  final String V_TWO_DECIMAL =  "^-?\\d+(\\.\\d{1,2})?$";

    private ValidateUtil(){}

    /**
     * 验证是不是整数
     * @param value 要验证的字符串 要验证的字符串
     * @return  如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isInteger(String value){
        return match(V_INTEGER,value);
    }

    /**
     * 验证是不是正整数
     * @param value 要验证的字符串
     * @return  如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isUnsignedInteger(String value){
        return match(V_Z_INDEX,value);
    }

    /**
     * 验证是不是负整数
     * @param value 要验证的字符串
     * @return  如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isNegativeInteger(String value){
        return match(V_NEGATIVE_INTEGER,value);
    }

    /**
     * 验证是不是数字
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isNumber(String value){
        return match(V_NUMBER,value);
    }
    /**
     * 验证是不是数字加中英文
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isZYNumber(String value){
        return match(V_ZY_NUM,value);
    }

    /**
     * 验证是不是正数
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isPositiveNumber(String value){
        return match(V_POSITIVE_NUMBER,value);
    }

    /**
     * 验证是不是负数
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isNegatineNumber(String value){
        return match(V_NEGATINE_NUMBER,value);
    }

    /**
     * 验证一个月的31天
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isThirtyOneDays(String value){
        return match(V_31DAYS,value);
    }

    /**
     * 验证是不是ASCII
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isASCII(String value){
        return match(V_ASCII,value);
    }


    /**
     * 验证是不是中文
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isChinese(String value){
        return match(V_CHINESE,value);
    }
    /**
     * 验证是不是颜色
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isColor(String value){
        return match(V_COLOR,value);
    }

    /**
     * 验证日期年月格式2017-07
     * @param value
     * @return
     */
    public static boolean isYearMonth(String value){
        return match(V_YEAR_MONTH,value);
    }


    /**
     * 验证是不是邮箱地址
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isEmail(String value){
        return match(V_EMAIL,value);
    }

    /**
     * 验证是不是浮点数
     * @param value 要验证的字符串
     * @return  如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isFloat(String value){
        return match(V_FLOAT,value);
    }

    /**
     * 验证是不是正确的身份证号码
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isIdNo(String value){
        return match(V_IDCARD,value);
    }

    /**
     * 验证是不是正确的IP地址
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isIp(String value){
        return match(V_IP4,value);
    }

    /**
     * 验证是不是字母
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isLetter(String value){
        return match(V_LETTER,value);
    }

    /**
     * 验证是不是小写字母
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isLowerCaseLetter(String value){
        return match(V_LETTER_I,value);
    }

    /**
     * 验证是不是大写字母
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isUpperCaseLetter(String value){
        return match(V_LETTER_U,value);
    }


    /**
     * 验证是不是手机号码
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isCellphone(String value){
        return match(V_PHONE,value);
    }

    /**
     * 验证是不是电话
     * @param value
     * @return
     */
    public static boolean isMobile(String value){
        return match(V_MOBILE,value);
    }

    /**
     * 验证是不是负浮点数
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isNegativeFloat(String value){
        return match(V_NEGATIVE_FLOAT,value);
    }

    /**
     * 验证非空
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isNotempty(String value){
        return match(V_NOTEMPTY,value);
    }

    /**
     * 验证密码的长度(6~18位)
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isNumberLength(String value){
        return match(V_P_LENGTH,value);
    }

    /**
     * 验证密码(数字和英文同时存在)
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isPasswordReg(String value){
        return match(V_P_REG,value);
    }

    /**
     * 验证图片
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isPicture(String value){
        return match(V_PICTURE,value);
    }

    /**
     * 验证正浮点数
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isPosttiveFloat(String value){
        return match(V_POSTTIVE_FLOAT,value);
    }

    /**
     * 验证QQ号码
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isQQnumber(String value){
        return match(V_QQ_NUMBER,value);
    }

    /**
     * 验证压缩文件
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isRar(String value){
        return match(V_RAR,value);
    }

    /**
     * 验证电话
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isTel(String value){
        return match(V_TEL,value);
    }

    /**
     * 验证非正浮点数
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isUnNegativeFloat(String value){
        return match(V_UN_NEGATIVE_FLOAT,value);
    }

    /**
     * 验证非负浮点数
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isUnpositiveFloat(String value){
        return match(V_UNPOSITIVE_FLOAT,value);
    }

    /**
     * 验证URL
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isUrl(String value){
        return match(V_URL,value);
    }

    /**
     * 验证用户注册。匹配由数字、26个英文字母或者下划线组成的字符串
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isUserName(String value){
        return match(V_USERNAME,value);
    }

    /**
     * 验证邮编
     * @param value 要验证的字符串
     * @return 如果是符合格式的字符串,返回 <b>true </b>,否则为 <b>false </b>
     */
    public static boolean isZipcode(String value){
        return match(V_ZIPCODE,value);
    }

    /**
     * 是否是价格
     * @param value
     * @return
     */
    public static boolean isPrice(String value){
        return match(V_PRICE,value);
    }

    /**
     * 判断是否金额格式
     * @param value
     * @return
     */
    public static boolean isMoney(String value){
        return match(V_MONEY,value);
    }

    public static boolean isPersonName(String value) {
        return match(V_PERSON_NAME,value);
    }

    public static boolean isMathOperator(String value) {
        return match(V_MATH_OPERATOR,value);
    }

    public static boolean isZYWord(String value) {
        return match(V_ZY,value);
    }

    public static boolean isTwoDecimal(Double value) {
        String str = value.toString();
        return match(V_TWO_DECIMAL,str);
    }

   private static boolean match(String regex, String str){
        Pattern pattern = Pattern.compile(regex);
        Matcher matcher = pattern.matcher(str);
        return matcher.matches();
    }

}
View Code

 

  • @Validated和@Valid的区别
  1. @Valid:标准JSR-303规范的标记型注解,用来标记验证属性和方法返回值,进行级联和递归校验
  2. @Validated:Spring的注解,是标准JSR-303的一个变种(补充),提供了一个分组功能,可以在入参验证时,根据不同的分组采用不同的验证机制
  3. 在Controller中校验方法参数时,使用@Valid和@Validated并无特殊差异(若不需要分组校验的话)
  4. @Validated注解可以用于类级别,用于支持Spring进行方法级别的参数校验。@Valid可以用在属性级别约束,用来表示级联校验
  5. @Validated只能用在类、方法和参数上,而@Valid可用于方法、字段、构造器和参数上