如何使用自定义注解进行参数校验

背景

在日常的开发工作过程中,我们用的比较多的参数校验注解有 @NotBlank、@NotBlank、@NotEmpty、@Min、@Max 等等。这些注解只实现了一些基本校验,但是实际工作当中有很多参数的校验要比这个复杂的多(例如:校验输入参数是否符合身份证的规则)。当我们面对这些复杂校验的时候,这些基本的注解就无能为力了。

通常的解决方案

如果工作中真遇到上述的情况,我们该怎么办呢?

通常的做法是将复杂的校验逻辑写在业务代码中,如下:

public class AccountController {
    public AccountVo getAccount(@RequestParam("cardNo") String cardNo) {
        if (!isValid(cardNo) {
            // 校验证件号是否有效,若无效则提示无效证件号
        }
        
        // TODO 验证通过后,进行业务处理
        ...
        
        return accountVo;
    }
}

这种实现方式的弊端是:参数校验逻辑与业务处理逻辑掺杂在一起,看起来并不是那么简洁。

优化后的解决方案

我们可以通过 「自定义注解 + 自定义验证器」将参数验证逻辑与业务处理逻辑剥离开。具体做法如下:

  1. 首先我们需要自定义一个 @IdCard 注解,代码如下:

    @Target({ ElementType.FIELD, ElementType.PARAMETER })
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface IdCard {
        String message() default “cardNo invalid”;
    
        Class<?>[] groups() default { };
    
        Class<? extends Payload>[] payload() default { };
    } 
    
  2. 然后,我们需要自定义一个针对 @IdCard 注解的验证器,代码如下:

    public class IdCardValidator implements ConstraintValidator<IdCard, String> {
        @Override
        public boolean isValid(String value, ConstraintValidatorContext context) {
            // 这里的 value 就是输入参数(证件号),例如:230102197701013467
    
            // TODO 根据自己的实际业务情况,编写校验逻辑,返回校验结果
            return true;
        }
    } 
    

    这里需要说明的是,如下图:
    在这里插入图片描述

  3. 当我们为 @IdCard 注解创建好了对应的自定义验证器 IdCardValidator 后,我们还要回头在 @IdCard 注解的定义类上标识 @IdCard 与 IdCardValidator 的关系,如下图:
    在这里插入图片描述
    如上图,需要在 @IdCard 注解的定义类上增加 @Constraint(validatedBy = IdCardValidator.class) 注解。

  4. 至此,我们就可以像使用 @NotNull 注解一样来使用自定义 @IdCard 注解了,代码如下:

     import org.springframework.validation.annotation.Validated;
     import org.springframework.web.bind.annotation.RestController;
     
     @RestController
     @Validated
     public class AccountController {
         public AccountVo getAccount(@RequestParam("cardNo") @IdCard String cardNo) {
            return accountVo;
         }
     }
    
  • 2
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
Java中的自定义注解是一种给代码提供额外信息的方式,可以在运行时通过反射机制获取注解的信息。通过自定义注解,我们可以实现参数校验,提高代码的健壮性和可维护性。 首先,我们需要定义一个注解类,用于定义参数校验的规则。比如,我们可以定义一个注解叫做@ParamCheck,用于对方法的参数进行校验。 ```java @Target(ElementType.PARAMETER) @Retention(RetentionPolicy.RUNTIME) public @interface ParamCheck { String value(); } ``` 接着,在需要进行参数校验的方法上使用@ParamCheck注解,并给注解传入校验规则的表达式。比如,我们可以给一个名为checkNumber的方法的参数添加校验注解。 ```java public void checkNumber(@ParamCheck("number > 0") int number) { // ... } ``` 然后,在方法内部,通过反射机制获取参数的注解信息,并根据注解中定义的校验规则对参数进行校验。 ```java public void checkNumber(int number) { Parameter parameter = ...; // 获取方法的参数信息 ParamCheck paramCheckAnnotation = parameter.getAnnotation(ParamCheck.class); if (paramCheckAnnotation != null) { String expression = paramCheckAnnotation.value(); // 根据expression对number进行校验 // ... } } ``` 最后,我们可以在调用checkNumber方法时传入一个不满足校验规则的参数,比如-10,当方法内部进行参数校验时,可以捕获到校验失败的情况,并进行相应处理。 ```java checkNumber(-10); // 参数校验失败,抛出异常或者进行其他处理 ``` 通过自定义注解实现参数校验可以方便地对代码进行统一的校验规则管理,提高代码的可维护性和可读性。同时,由于注解是在运行时通过反射获取,可以对代码进行动态改变和扩展,使得我们可以更加灵活地进行参数校验
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

cab5

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值