自定义注解进行入参校验,附代码及原理浅析

自定义注解

 

项目开放接口的时候通常需要对接口入参进行一定的校验,此文章拿最常见的非空验证进行举例。


大致的原理如下:定义一个自定义注解,对自定义注解进行拦截进行校验,不满足条件抛出自定义异常,捕获自定义异常进行处理并返回错误信息给调用者。


首先定义一个Dto进行入参的接收

public class WingzingDto implements Serializable {

    private String userId;

    private String userName;

    //get set

}

配置一个自定义注解类

/**
 * 接口参数非空校验注解
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface ApiParamValidate {
    String value();
}

然后编写一个切面类,对此注解进行拦截。

/**
 * 接口入参非空验证,切面处理类
 *
 */
@Aspect
public class ApiParamValidateAspect {

    @Pointcut("@annotation(com.xx.xx.ApiParamValidate)")
    public void apiParamValidatePointCut() {

    }

    @Around("apiParamValidatePointCut()")
    public Object around(ProceedingJoinPoint point) throws Throwable {
        MethodSignature signature = (MethodSignature) point.getSignature();
        Method method = signature.getMethod();

        ApiParamValidate apiParamValidate = method.getAnnotation(ApiParamValidate.class);
        if (apiParamValidate != null) {
            String requestParam = apiParamValidate.value();
            if(requestParam.length()>0){
                String [] paramArray = requestParam.split(",");//注解配置的所需校验参数
                Object[] paramValues = point.getArgs();//api接口定义的参数
                String[] paramValuesNames = signature.getParameterNames();//api接口定义的参数名称
                if(paramValues.length>0){
                    for(String needParamName : paramArray){
                        validate(needParamName,paramValues,paramValuesNames);
                    }
                }else{
                    //参数为空
                    throw new RRException("参数与api接口不匹配!");
                }
            }else{
                throw new RRException("api接口若不需要校验参数,请不要使用此注解!");
            }
        }
        return point.proceed();
    }

    private static boolean isJavaClass(Class<?> clz) {
        return clz != null && clz.getClassLoader() == null;
    }

    private static void validate(String needParamName,Object[] objs,String[] objNames){
        if(objs!=null&&objs.length>0){
            boolean b = false;
            for(int i=0;i<objs.length;i++){
                if(objs[i]!=null){
                    if(!isJavaClass(objs[i].getClass())){//参数是自定义对象
                        //自定义类
                        Class clz = objs[i].getClass();
                        try{
                            Method mt = clz.getMethod("get"+needParamName.substring(0,1).toUpperCase()+needParamName.substring(1));
                            Object ret = mt.invoke(objs[i]);
                            if(ret==null||ret.toString().equals("")){
                                throw new RRException("参数 "+needParamName+" 为空");
                            }else{
                                b = true;//在参数中找到了必填参数,且值不为空
                            }
                        }catch (NoSuchMethodException e){
                            System.out.println("没有在此对象中找到参数 "+needParamName+" 的getter setter方法");
                        }catch(Exception e){
                            System.out.println("没有在此对象中找到参数 "+needParamName+" 的getter setter方法");
                        }
                    }else{//参数是系统类
                        if(objNames[i].equals(needParamName)){
                            if(objs[i]==null||objs[i].toString().equals("")){
                                throw new RRException("参数 "+needParamName+" 为空");
                            }else{
                                //在参数中找到了必填参数,且值不为空
                                b = true;
                            }
                        }
                    }
                }
            }
            if(!b){
                //没有找到必填参数
                throw new RRException("参数 "+needParamName+" 为空");
            }
        }else{
            throw new RRException("api接口入参未定义!请联系管理员!");
        }
    }

}
@Aspect 用来标识此类是一个切面供容器进行读取
@Pointcut 切点 用于标识什么时候进行拦截,后面的@annotation标识了具体拦截的类 即我定义的自定注解
@Around 环绕增强,与@AfterReturning,@Before,@AfterThrowing,@After等等类似。

这里实现的效果就是对自定义注解进行拦截,然后在 around方法中进行了参数循环调用validate方法,通过反射的方式进行调用get方法,然后判断是否为空(这里也可以进行各种各样的校验)。

我选择抛出自定义的异常RRException,来提示调用接口方错误信息。

附上RRException,以及对应的捕获该异常的类。

/**
 * 自定义异常
 *
 */
public class RRException extends RuntimeException {
    private static final long serialVersionUID = 1L;

    private String msg;
    private int code = 500;

    public RRException(String msg) {
        super(msg);
        this.msg = msg;
    }

    public RRException(String msg, Throwable e) {
        super(msg, e);
        this.msg = msg;
    }

    public RRException(String msg, int code) {
        super(msg);
        this.msg = msg;
        this.code = code;
    }

    public RRException(String msg, int code, Throwable e) {
        super(msg, e);
        this.msg = msg;
        this.code = code;
    }

    public String getMsg() {
        return msg;
    }

    public void setMsg(String msg) {
        this.msg = msg;
    }

    public int getCode() {
        return code;
    }

    public void setCode(int code) {
        this.code = code;
    }


}
/**
 * 异常处理器
 *
 */
@ControllerAdvice
public class RRExceptionHandler {
    private Logger logger = LoggerFactory.getLogger(getClass());

    /**
     * 处理自定义异常
     */
    @ExceptionHandler(RRException.class)
    @ResponseBody
    public String handleRRException(RRException e){
        logger.debug(e.getMessage(), e);
        System.out.println(e.toString());
        return BackResult.error(e.getMsg()).toJson();
    }

    @ExceptionHandler(NoHandlerFoundException.class)
    @ResponseBody
    public String handlerNoFoundException(NoHandlerFoundException e) {
        logger.error(e.getMessage(), e);
        e.printStackTrace();
        return BackResult.error("路径不存在,请检查路径是否正确").toJson();
    }

    @ExceptionHandler(Exception.class)
    @ResponseBody
    public String handleException(Exception e){
        logger.error(e.getMessage(), e);
        e.printStackTrace();
        return BackResult.error("未知异常,请联系管理员").toJson();
    }
}
@ControllerAdvice 增强型controller,使用此注解可以使得整个项目的异常都可以被此捕获。
@ExceptionHandler 异常捕获,后面的参数代表拦截什么类型的异常。

 使用postman调用看一下大概的效果:

  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是一个自定义注解校验身份证的代码示例: 首先,我们需要定义一个注解 `@IdNumber`,用于标注需要进行身份证校验的字段: ```java @Documented @Constraint(validatedBy = IdNumberValidator.class) @Target({ElementType.FIELD, ElementType.PARAMETER}) @Retention(RetentionPolicy.RUNTIME) public @interface IdNumber { String message() default "身份证号码不正确"; Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {}; } ``` 其中,`@Constraint(validatedBy = IdNumberValidator.class)` 表示该注解需要使用 `IdNumberValidator` 进行校验。 接着,我们需要实现 `IdNumberValidator`: ```java public class IdNumberValidator implements ConstraintValidator<IdNumber, String> { @Override public void initialize(IdNumber constraintAnnotation) { } @Override public boolean isValid(String value, ConstraintValidatorContext context) { if (StringUtils.isBlank(value)) { return true; } return isIdNumber(value); } /** * 身份证校验 * * @param idNumber 身份证号码 * @return 是否有效 */ private boolean isIdNumber(String idNumber) { String regex = "^\\d{15}$|^\\d{17}[0-9Xx]$"; if (!idNumber.matches(regex)) { return false; } String provinceCode = idNumber.substring(0, 2); if (!ProvinceCodeUtils.isValidProvinceCode(provinceCode)) { return false; } if (!isValidBirthDate(idNumber)) { return false; } return isValidCheckCode(idNumber); } /** * 校验生日日期 * * @param idNumber 身份证号码 * @return 是否有效 */ private boolean isValidBirthDate(String idNumber) { String birthDateStr; if (idNumber.length() == 15) { birthDateStr = "19" + idNumber.substring(6, 12); } else { birthDateStr = idNumber.substring(6, 14); } SimpleDateFormat format = new SimpleDateFormat("yyyyMMdd"); format.setLenient(false); try { format.parse(birthDateStr); } catch (ParseException e) { return false; } return true; } /** * 校验校验码 * * @param idNumber 身份证号码 * @return 是否有效 */ private boolean isValidCheckCode(String idNumber) { if (idNumber.length() == 15) { return true; } char[] idCharArray = idNumber.toCharArray(); int[] idIntArray = new int[17]; for (int i = 0; i < 17; i++) { idIntArray[i] = Integer.parseInt(String.valueOf(idCharArray[i])); } int[] weightFactorArray = new int[]{7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2}; int sum = 0; for (int i = 0; i < 17; i++) { sum += idIntArray[i] * weightFactorArray[i]; } int checkCodeIndex = sum % 11; char[] checkCodeArray = {'1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'}; char checkCode = checkCodeArray[checkCodeIndex]; return checkCode == idCharArray[17]; } } ``` 其中,`isValid` 方法用于进行身份证校验,`isValidBirthDate` 方法用于校验生日日期,`isValidCheckCode` 方法用于校验校验码。 最后,在需要进行身份证校验的字段上使用 `@IdNumber` 注解即可: ```java public class User { @IdNumber private String idNumber; // getter/setter 略 } ``` 这样,在使用 `@Valid` 注解校验实体时,如果 `idNumber` 字段不符合身份证规则,则会抛出 `ConstraintViolationException` 异常。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值