java自定义注解验证手机格式

java自定义注解验证手机格式

1、@Valid与@Validated的区别

1.1 基本区别

  • @Valid:Hibernate validation校验机制

  • @Validated:Spring Validator校验机制,这个也是最常用的

  • @Validation只是对@Valid进行了二次封装,在使用上并没有太大区别,但在分组、注解位置、嵌套验证等功能上有所不同

1.2 作用范围

  • @Validated:用在类型、方法和方法参数上。但不能用于成员属性(field)
  • @Valid:可以用在方法、构造函数、方法参数和成员属性(field)上

1.3 分组校验

  • @Validated:提供分组功能,可以在参数验证时,根据不同的分组采用不同的验证机制,注解中必须提供groups属性,该属性就是做分组的必要参数
  • @Valid:没有分组功能

2、未使用分组校验的示例

注解:

/**
 * 手机号验证正则
 */
@Target({ElementType.FIELD,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Constraint(validatedBy = {PhoneValidator.class})// 指定约束处理器,也就是手机号格式验证是哪个类来做校验
public @interface Phone {

    String pattern() default "^(?:(?:\\+|00)86)?1\\d{10}$";

    String message() default "手机号格式非法";

    Class<?>[] groups() default { }; // groups用来指定分组,可以让校验采取不同的机制,当前默认未指定任何分组机制,默认每次都要进行校验

    Class<? extends Payload>[] payload() default { };
    
    // 默认分组
    interface Default{

    }

    // 分组A
    interface A{

    }

}

格式校验处理器:

/**
 * 校验处理器:做手机号码格式验证的核心类
 */
public class PhoneValidator implements ConstraintValidator<Phone, String> {

    // 注解对象
    private Phone phone;

    // 初始化【Phone】对象
    @Override
    public void initialize(Phone constraintAnnotation) {
        phone = constraintAnnotation;
    }

    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {
        // 获取【Phone】对象的手机格式验证表达式
        String pattern = phone.pattern();
        Pattern compile = Pattern.compile(pattern);
        Matcher matcher = compile.matcher(value);
        return matcher.matches();
    }

作用类:

@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
public class Person implements Serializable {
    
    @Phone
    private String phone;

}

注意:只有在spring或者springboot项目中才能使用,直接调用方法不会有任何效果,使用注解进行对象的属性格式校验时,必须配合@Validated一起使用(不一起使用,格式校验注解将会无效),正确操作如下:

@RestController
@RequestMapping("/admin/")
public class PersonController {

    @Autowired
    private PersonService personService;

    @PostMapping("/query")
    public Person query(@RequestBody @Validated Person params) {
        return JsonResult.success(personService.queryByPhone(params));
    }
}

以上示例未使用分组功能,因此每次都会校验。

3、分组校验的示例

使用分组校验示示例时,先要看看@Validated注解,因为分组校验就是配合该注解一起使用的,通过阅读注释就能理解到value属性就是用来指定分组的:

@Target({ElementType.TYPE, ElementType.METHOD, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Validated {

	/**
	 * Specify one or more validation groups to apply to the validation step
	 * kicked off by this annotation.
	 * <p>JSR-303 defines validation groups as custom annotations which an application declares
	 * for the sole purpose of using them as type-safe group arguments, as implemented in
	 * {@link org.springframework.validation.beanvalidation.SpringValidatorAdapter}.
	 * <p>Other {@link org.springframework.validation.SmartValidator} implementations may
	 * support class arguments in other ways as well.
	 */
	Class<?>[] value() default {};

}

因此我们需要改动的位置有两处:

  • 首先是注解的作用类,注解上指定groups属性
  • 其次是controller中的请求的形参:在请求中形参的@Validated指定value值,也就是指定校验生效的分组,如果请求中的分组类型【@Validatedvalue值】和作用类中注解所指定的分组【@Phone中的groups属性的值】一致时,才会进行校验,否则不会执行校验

作用类:

@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
public class Person implements Serializable {
    
    // 指定groups属性
    @Phone(groups = {Phone.A.class})
    private String phone;

}

controller层:

@RestController
@RequestMapping("/admin/")
public class PersonController {

    @Autowired
    private PersonService personService;

    @PostMapping("/query")
    public Person query(@RequestBody @Validated(Phone.A.class) Person params) {
        return JsonResult.success(personService.queryByPhone(params));
    }
}

此时请求中的校验分组Phone.A.class和作用类中的校验分组Phone.A.class一致,所以校验会被执行

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
要使用自定义注解判断数据类型,你可以按照以下步骤进行操作: 1. 首先,创建一个自定义注解,用于标记需要检查数据类型的元素。例如: ```java import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.FIELD) public @interface DataTypeCheck { String dataType(); } ``` 在上述示例中,我们创建了一个名为 `DataTypeCheck` 的自定义注解,并使用 `dataType` 属性来指定要检查的数据类型。 2. 接下来,在需要进行数据类型检查的地方,使用该注解标记相应的字段。例如: ```java public class MyClass { @DataTypeCheck(dataType = "String") private String name; @DataTypeCheck(dataType = "int") private int age; // 其他字段... } ``` 在上述示例中,我们在 `name` 字段和 `age` 字段上使用了 `@DataTypeCheck` 注解,并分别指定了要检查的数据类型。 3. 最后,编写一个用于检查数据类型的工具类或方法。该工具类或方法可以通过反射获取字段上的注解,并根据注解中指定的数据类型进行判断。例如: ```java import java.lang.reflect.Field; public class DataTypeChecker { public static boolean check(Object obj) throws IllegalAccessException { Class<?> clazz = obj.getClass(); Field[] fields = clazz.getDeclaredFields(); for (Field field : fields) { if (field.isAnnotationPresent(DataTypeCheck.class)) { DataTypeCheck annotation = field.getAnnotation(DataTypeCheck.class); String expectedType = annotation.dataType(); Object value = field.get(obj); String actualType = value.getClass().getSimpleName(); if (!expectedType.equals(actualType)) { System.out.println("数据类型错误:" + field.getName()); return false; } } } return true; } } ``` 在上述示例中,我们定义了一个名为 `DataTypeChecker` 的工具类,其中的 `check` 方法可以检查对象中带有 `@DataTypeCheck` 注解的字段是否符合指定的数据类型。 你可以在需要检查数据类型的地方调用 `DataTypeChecker.check` 方法来进行验证。例如: ```java public class Main { public static void main(String[] args) throws IllegalAccessException { MyClass obj = new MyClass(); obj.name = "John"; obj.age = 25; boolean isDataTypeValid = DataTypeChecker.check(obj); System.out.println("数据类型是否有效:" + isDataTypeValid); } } ``` 在上述示例中,我们创建了一个 `MyClass` 对象,并设置了 `name` 和 `age` 字段的值。然后,我们调用 `DataTypeChecker.check` 方法来检查数据类型是否有效,并输出结果。 这样,你就可以使用自定义注解和反射来判断数据类型了。希望对你有所帮助!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值