实现ConstraintValidator完成自定义校验注解

本文介绍了如何在Spring Boot中创建一个自定义注解`OnlyNumber`,用于验证字符串是否只包含数字并且长度在指定范围内。注解包含`min`和`max`属性以设定限制,并在`OnlyNumberClass`中实现`ConstraintValidator`进行逻辑校验。在实际使用中,需将注解应用于字段并配合`@Valid`注解在Controller层进行参数校验。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、导包

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-validation</artifactId>
</dependency>

二、自定义注解类

此处我用一个非常简单的例子来说明,主要是看结构。
OnlyNumber :

/**
 * @Description: 只能包含数字,同时可以控制数字位数
 * @Author: Felix.Du
 * @Date: 2022/3/8 7:01 下午
 */
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD,ElementType.PARAMETER})
@Constraint(validatedBy = OnlyNumberClass.class)
public @interface OnlyNumber {

    int min() default 1;

    int max() default 99;

    String message() default "VERIFICATION_CODE_ERROR";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};
}

注解中的变量可以自己控制,我这里定义了min(最小位数),max(最大位数)两个。其中message为不满足要求提示的错误信息。
接下来我们来看OnlyNumberClass类:

/**
 * @Description: 只能为数字类型的String,且字符长度介于[min,max]之间
 * @Author: Felix.Du
 * @Date: 2022/3/9 10:55
 */
public class OnlyNumberClass implements ConstraintValidator<OnlyNumber, Object> {

    private int min;
    private int max;

    private static final String REGEX = "^[-\\+]?[\\d]*$";

    @Override
    public void initialize(OnlyNumber onlyNumber) {
        this.min = onlyNumber.min();
        this.max = onlyNumber.max();
    }

    @Override
    public boolean isValid(Object value, ConstraintValidatorContext constraintValidatorContext) {
        if (value == null) {
            return true;
        }
        //判断是否为纯数字
        Pattern pattern = Pattern.compile(REGEX);
        boolean isNumber = pattern.matcher(value.toString()).matches();

        //判断位数是否合法
        int length = value.toString().trim().length();
        boolean isUp =  min <= length;
        boolean isDown = max >= length;

        return isUp && isDown && isNumber;
    }
}

实现ConstraintValidator类主要是重写两个方法。
1.initialize(),会在校验实例化后被调用,一般用于做些初始化工作。
2.isValid(),实际执行验证的方法,第一个参数获取的是字段或对象实际对应的值,取决于类的枚举限定类型。第二个参数是ConstraintValidatorContext,上下文可以做些默认的设置。
我们主要是在isValid()方法中写自己的逻辑。其中返回true为校验通过,返回false为校验失败,给出错误信息message。

三、实际使用

通常我们的字段是放在对象里的,这个时候我们如果只在字段上打上这个注解通常是不会生效的。如:

@Data
@NoArgsConstructor
@AllArgsConstructor
public class BindEmailOrPhoneParam {

    /**
     * 验证码
     */
    @ApiModelProperty(value = "验证码", required = true)
    @NotNull(message = "NOT_NULL")
    @OnlyNumber(min = 5, max = 5)
    private String code;

}

我们还需要在Controller层具体方法的参数前加上@Valid 注解,这个时候它才会认为我们需要校验这个对象里面的参数,如:

/**
 * 绑定手机、邮箱
 *
 * @param userId    用户id
 * @param param     绑定邮箱、手机请求参数
 * @return          0->失败, 1->成功
 */
@ApiOperation(value = "绑定手机、邮箱", notes = "绑定手机、邮箱", httpMethod = "POST")
@PostMapping(value = "/bind/account/{userId}", produces = MediaType.APPLICATION_JSON_VALUE)
public RestResult<Integer> appBindEmailOrPhone(@PathVariable("userId") Long userId,
                                        @RequestBody @Valid BindEmailOrPhoneParam param) {
    return RestResult.ok(RpcClientUtil.sync(() ->
                            tmdUserManageClient.bindEmailOrPhone(userId, BeanUtil.toBean(param, BindEmailOrPhoneDTO.class))
            )
    );
}
### Java自定义注解实现参数校验的功能 在Java开发中,通过自定义注解可以灵活地满足复杂的业务需求。以下是基于提供的引用内容以及标准实践的一个完整示例。 #### 创建自定义注解 首先需要定义一个注解类,并指定其作用范围和生命周期。以下是一个名为`@State`的自定义注解: ```java package com.example.validate; import javax.validation.Constraint; import javax.validation.Payload; import java.lang.annotation.*; @Documented @Target({ElementType.METHOD, ElementType.FIELD, ElementType.PARAMETER}) @Retention(RetentionPolicy.RUNTIME) @Constraint(validatedBy = StateValidator.class) // 验证器类 public @interface State { String value(); // 定义允许的状态值列表 String message() default "Invalid state"; // 默认错误提示信息 Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {}; } ``` 此部分展示了如何创建一个简单的自定义注解[@State],它能够接收一组合法状态作为输入并提供默认的消息处理机制[^1]。 #### 编写验证逻辑 接着需编写具体的验证逻辑,即实现`javax.validation.ConstraintValidator<A,T>`接口。这里以检查整数类型的字段是否属于预设的一组有效状态为例: ```java package com.example.validate; import javax.validation.ConstraintValidator; import javax.validation.ConstraintValidatorContext; import java.util.Arrays; import java.util.List; public class StateValidator implements ConstraintValidator<State, Integer> { private List<Integer> allowedStates; @Override public void initialize(State constraintAnnotation) { this.allowedStates = Arrays.asList(constraintAnnotation.value().split(",")) .stream() .map(Integer::parseInt) .toList(); } @Override public boolean isValid(Integer value, ConstraintValidatorContext context) { if (value == null || allowedStates.contains(value)) { return true; } return false; } } ``` 上述代码片段实现了对传入数值的合法性判断过程,其中初始化方法负责解析配置好的可接受状态集合;而有效性检测函数则依据当前实例属性决定是否符合预期条件[^2]。 #### 使用自定义注解 最后可以在实体类或者控制器层的方法签名上运用该注解来进行约束声明: ```java public class OrderRequest { @State("0,1") // 只能取值为0或1 private int orderStatus; // Getter and Setter omitted... } // 或者应用于Controller中的请求参数 @RestController @RequestMapping("/orders") public class OrderController { @PostMapping("/{id}") public ResponseEntity<String> updateOrder(@PathVariable Long id, @RequestBody @Valid OrderRequest request){ // Business logic here ... return ResponseEntity.ok("Success"); } } ``` 以上例子说明了怎样利用Spring框架下的自动注入特性配合Hibernate Validator完成基本的数据完整性检验工作流程^。 --- 问题
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值