@param注解的用法解析_Spring AOP 使用注解进行校验实例(valid)

点击上方“Java知音”,选择“置顶公众号”

技术文章第一时间送达!

作者:CatalpaFlat

juejin.im/post/5a6301e051882573315c769d

先说明一下@PathVariable和@RequestParam两个注解的用法。

1.原版本讲解

1.1 @PathVariable

1.1.1 RESTful风格

格式:path/1/catalpaFlat eg:

@GetMapping("path/{isInt}/{isString}")
public ResponseVO pathGet(@PathVariable Integer isInt,
                          @PathVariable String isString) {
   log.info("int:" + isInt);
   log.info("String:" + isString);
  JSONObject resultJson = new JSONObject();
  resultJson.put("isInt", isInt);
  resultJson.put("isString", isString);
  return new ResponseVO(HttpStatus.OK.value(), "pathGet", resultJson);
}

request:http://localhost:8888/path/3/dadas

1.1.2 校验

代码式校验

if(isInt 2){
  return new ResponseVO(HttpStatus.BAD_REQUEST.value(), "pathGet", "isInt must be more than 2");
}

代码式校验的话,个人觉得就有点儿不必要,除非是一些特殊需求

1.2 @RequestParam

1.2.1 表单提交(query)

格式:query?isInt=2&isString=catalpaFlat

@GetMapping("query?")
public ResponseVO queryGet(@RequestParam Integer isInt,
                           @RequestParam String isString) {
    log.info("int:" + isInt);
    log.info("String:" + isString);
    JSONObject resultJson = new JSONObject();
    resultJson.put("isInt", isInt);
    resultJson.put("isString", isString);
    return new ResponseVO(HttpStatus.OK.value(), "queryGet", resultJson);
}
1.2.2 校验

同样也需要代码式校验

if(isInt 2){
  return new ResponseVO(HttpStatus.BAD_REQUEST.value(), "queryGet", "isInt must be more than 2");
}

因为@Valid + BindingResult只能用于@ResponseBody这种类型注解。

2. AOP改良版

可以感受到原版本都得在代码中进行校验,只要使用到@PathVariable和@RequestParam的都得进行一波参数校验,因此就想能不能像@Valid一样,为其添加注解式校验,这样代码量就减少并让代码优雅了很多。

2.1 接口层(IDAL)

以下代码再则是改良之后的代码:

  • @ParameterValid 相当于@Valid,并且在其属性中进行配置参数校验规则

  • @PathAndQueryParamValid 相当于AOP的切入点

@PathAndQueryParamValid
@GetMapping("path/{isInt}/{isString}")
public ResponseVO pathGet(@PathVariable @ParameterValid(type = Integer.class, msg = "isInt must be more than 2", isMin = true, min = 2) Integer isInt,
                          @PathVariable @ParameterValid(type = String.class, msg = "isString is empty") String isString) {
    log.info("int:" + isInt);
    log.info("String:" + isString);
    JSONObject resultJson = new JSONObject();
    resultJson.put("isInt", isInt);
    resultJson.put("isString", isString);
    return new ResponseVO(HttpStatus.OK.value(), "pathGet", resultJson);
}
@GetMapping("query")
@PathAndQueryParamValid
public ResponseVO queryGet(@RequestParam @ParameterValid(type = Integer.class, msg = "isInt must be more than 2 ", isMin = true, min = 2) Integer isInt,
                          @RequestParam @ParameterValid(type = String.class, msg = "isString is empty") String isString) {
   log.info("int:" + isInt);
   log.info("String:" + isString);
   JSONObject resultJson = new JSONObject();
   resultJson.put("isInt", isInt);
   resultJson.put("isString", isString);
   return new ResponseVO(HttpStatus.OK.value(), "queryGet", resultJson);
}

2.2 自定义注解(annotation)

2.2.1 @PathAndQueryParamValid

只是简单的用于方法类型注解:

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public  @interface  PathAndQueryParamValid {
}
2.2.2 @ParameterValid

@ParameterValid 可以根据实际业务需求添加属于你的校验规则:

@Documented
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface ParameterValid {
    Class> type();

    String msg();

    boolean request() default true;

    boolean isEmpty() default true;

    boolean isBlank() default true;

    boolean isNull() default false;

    int min() default 0;
    int max() default 0;
    int[] section() default {0,0};
    boolean isMin() default false;
    boolean isMax() default false;
    boolean isSection() default false;
}

2.3 AOP切面(重点-1)

  • 通过joinPoint获取切点方法名以及类名,后续(重点)有大用

  • 通过JoinPoint获取方法的参数

  • 调用(重点2)ParamValidSupport

@Aspect
@Component
public class PathAndQueryParamValidAspect {

  private static final Logger log = LoggerFactory.getLogger(PathAndQueryParamValidAspect.class);

  @Before("@annotation(paramValid)")
  public void paramValid(JoinPoint joinPoint, PathAndQueryParamValid paramValid) {
      String className = joinPoint.getTarget().getClass().getName();
      String methodName = joinPoint.getSignature().getName();
      Object[] param = joinPoint.getArgs();
      try {
          List<String> errorLists = ParamValidSupport.get().validate(className, methodName,
                  ParameterValid.class, param);
          if (errorLists != null) {
              AdvanceResponseSupport.advanceResponse(
                      new ResponseVO(HttpStatus.BAD_REQUEST.value(), "parameter empty", errorLists));
          }
      } catch (NotFoundException | NoSuchMethodException | ClassNotFoundException e) {
          log.error("e-name:" + e.getClass().getName() + ":message:" + e.getMessage());
      }
  }
}

2.4 校验(重点-2)

  • 通过方法名和类名,根据ClassPool获取CtClass和CtMethod

  • 从而获取参数注解,解析参数注解规则进行校验参数

(这里还可以重构的,只是对int和string两种类型进行校验,还可以添加其他需求。把主要内容先呈现出来)

public class ParamValidSupport {
private static final Logger logger = LoggerFactory.getLogger(ParamValidSupport.class);

private static final String PARAM_TYPE_ERROR = "param type error";
private static final String INT_PARAM_ERROR = "Invalid interva";
private static final int INT_PARAM_TYPE_MAX_SIZE = 2;
private static final int INT_PARAM_SIZE_SUBSCRIPT_MIN = 0;
private static final int INT_PARAM_SIZE_SUBSCRIPT_MAX = 0;

private static final int STRING_SIZE = 2;
private static final char STRING_TYPE_END = '}';
private static final char STRING_TYPE_BEGIN = '{';
private static final char STRING_EMPTY_DOUBLE_CHARACTER = '"';
private static final char STRING_EMPTY_SINGLE_CHARACTER = '\'';

private static ParamValidSupport mInstance;

private ParamValidSupport() {
}

public static ParamValidSupport get() {
    if (mInstance == null) {
        synchronized (ParamValidSupport.class) {
            if (mInstance == null) {
                mInstance = new ParamValidSupport();
            }
        }
    }
    return mInstance;
}

 /**
 * 校验
 */
public List validate(String className, String methodName,
                             Class> annotationClass, Object[] args)
        throws NotFoundException, NoSuchMethodException, ClassNotFoundException {

    if (StringUtils.isBlank(className)) {
        return null;
    }
    if (StringUtils.isBlank(methodName)) {
        return null;
    }
    if (annotationClass == null) {
        return null;
    }

    ClassPool pool = ClassPool.getDefault();
    CtClass ct = pool.get(className);
    CtMethod ctMethod = ct.getDeclaredMethod(methodName);
    Object[][] parameterAnnotations = ctMethod.getParameterAnnotations();

    List errorLists = new ArrayList<>();for (int i = 0; i         Object[] parameterAnnotation = parameterAnnotations[i];
        Object param = args[i];for (Object object : parameterAnnotation) {
            Annotation annotation = (Annotation) object;
            Class extends Annotation> aClass = annotation.annotationType();if (aClass.equals(annotationClass)) {
                boolean isEmpty = ((ParameterValid) object).isEmpty();if (isEmpty) {
                    ParameterValid parameterValid = (ParameterValid) object;
                    String errorMsg = parameterValid.msg();if (Integer.class.isAssignableFrom(param.getClass())){int paramInt = (int) param;if (parameterValid.isMin() && paramInt                             errorLists.add(errorMsg);
                        }if (parameterValid.isMax() && paramInt                             errorLists.add(errorMsg);
                        }if (parameterValid.isSection()) {int[] section = parameterValid.section();if (section.length != INT_PARAM_TYPE_MAX_SIZE) {
                                logger.error(INT_PARAM_ERROR);throw new ParameterValidException(INT_PARAM_ERROR);
                            }if (!(paramInt > section[INT_PARAM_SIZE_SUBSCRIPT_MIN] && paramInt                                 errorLists.add(errorMsg);
                            } else if (!(paramInt > section[INT_PARAM_SIZE_SUBSCRIPT_MAX] && paramInt                                 errorLists.add(errorMsg);
                            }
                        }
                    }if (String.class.isAssignableFrom(param.getClass())){
                        String paramStr = (String) param;if (parameterValid.isNull()) {if (StringUtils.isEmpty(paramStr)) {
                                errorLists.add(errorMsg);
                            }
                        } else {if (parameterValid.isBlank()) {if (StringUtils.isBlank(paramStr)) {
                                    errorLists.add(errorMsg);
                                } else {int length = paramStr.length();char begin = paramStr.charAt(0);char end = paramStr.charAt(length - 1);if (STRING_TYPE_BEGIN == begin &&
                                            STRING_TYPE_END == end) {
                                        errorLists.add(errorMsg);
                                    }if (length == STRING_SIZE && STRING_EMPTY_DOUBLE_CHARACTER == begin
                                            && STRING_EMPTY_DOUBLE_CHARACTER == end) {
                                        errorLists.add(errorMsg);
                                    }if (length == STRING_SIZE && STRING_EMPTY_SINGLE_CHARACTER == begin
                                            && STRING_EMPTY_SINGLE_CHARACTER == end) {
                                        errorLists.add(errorMsg);
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    }if (errorLists.size() != 0) {return errorLists;
    }return null;
}
}

2.4 测试结果

521b55f45a6a3c7ec6e171f734e80ce7.png

92f6f4f7607d884f82018ccd6be58396.png

15508990d09a372490e55e63fd16f088.png

be142a444c2819de54d1e8a29660dafd.png

END

Java面试题专栏

【30期】说一下HashMap的实现原理? 【29期】Java集合框架 10 连问,你有被问过吗? 【28期】ZooKeeper面试那些事儿 【27期】Dubbo面试八连问,这些你都能答上来吗? 【26期】如何判断一个对象是否存活?(或者GC对象的判定方法)? 【25期】这三道常见的面试题,你有被问过吗? 【24期】请你谈谈单例模式的优缺点,注意事项,使用场景 【23期】请你谈谈关于IO同步、异步、阻塞、非阻塞的区别 【22期】为什么需要消息队列?使用消息队列有什么好处? 【21期】你能说说Java中Comparable和Comparator的区别吗

8e0d660ae19f2d034bd7e2a8328a34fe.png

我知道你 “在看ddc24d641207cc4fdad55193c4d4eedb.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值