Java增加自定义注解进行校验入参详解
作者:奔跑的毛球 发布时间:2023-01-05 13:34:25
背景
客户使用我们系统的时候,查询不带任何查询条件,查询就返回全部数据,500多万条数据啊,然后直接导出,数据量庞大,接口超时,这可苦了我们这些开发人员,一边优化,一边挨喷。这么多数据就算导成功了,Excel也打不开呀。迫不得已,决定强制让客户至少传入一个参数进行查询来缓解服务器以及开发人员的压力。
首先想到的,最简单的,就是增加一个静态方法,在每个方法入口调一下,来校验以及抛出错误。但是转念一想,更优美的解决方案是在调用的方法上加一个注解,使用注解来完成这个功能,这岂不是很棒。
再一句话说下需求:
增加注解对入参进行校验,保证至少有一个参数不为空,若是有时间参数,则起始时间和结束时间之间的距离不能超过30天。
接下来,Show Time
注解类
这里有三个参数,分别是三个参数名称,起始时间参数名称,结束时间参数名称,需要校验的参数名称
@Target({ElementType.PARAMETER, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface VerifyParameters {
/**
* 起始时间参数名称
*/
String startTimeParamName() default "startTime";
/**
* 结束时间参数名称
*/
String endTimeParamName() default "endTime";
/**
* 需要校验的参数名称
*/
String paramName() default "";
}
注解的Aspect类
这里贴一个完整的,目的是有的小伙伴想用的话,贴过去就能用。
@Component
@Aspect
public class VerifyParametersAspect {
private static final Logger logger = LoggerFactory.getLogger(VerifyParametersAspect.class);
/**
* 切点
*/
@Pointcut("@annotation(com.guava.mall.app.annotation.VerifyParameters)")
public void serviceAspect() {
}
/**
* service 方法前调用
*
* @param joinPoint
*/
@Before("serviceAspect()")
public void doBeforeService(JoinPoint joinPoint) {
try {
//获取方法参数名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
//获取方法
Method method = signature.getMethod();
//获取参数名
LocalVariableTableParameterNameDiscoverer u = new LocalVariableTableParameterNameDiscoverer();
String[] parameterNames = u.getParameterNames(method);
Map<String, Object> params = new HashMap<>(8);
params = getParamMap(joinPoint, method, parameterNames, params);
//获取注解
VerifyParameters verifyParameters = method.getAnnotation(VerifyParameters.class);
//参数名
String paramName = verifyParameters.paramName();
Object o = params.get(paramName);
ValidationUtils.validate(o == null, "参数不能为空");
ValidationUtils.validate(!atLeastOnePropertyNotNull(o), "请至少输入一个查询条件进行查询和导出");
//开始时间和结束时间的参数名
String s = verifyParameters.startTimeParamName();
String e = verifyParameters.endTimeParamName();
Map<?, ?> map = JSONUtils.bean2Map(o);
Object startTime = map.get(s);
Object endTime = map.get(e);
if (startTime != null || endTime != null) {
ValidationUtils.validate(startTime == null || endTime == null, "开始时间和结束时间必须同时存在");
ValidationUtils.validate(Integer.parseInt(String.valueOf(endTime)) - Integer.parseInt(String.valueOf(startTime)) > 30 * 24 * 60 * 60, "时间间隔不能超过一个月");
}
} catch (NumberFormatException ex) {
logger.error(ex.getMessage(), ex);
}
}
private Map<String, Object> getParamMap(JoinPoint joinPoint, Method method, String[] parameterNames, Map<String, Object> params) {
int i = 0;
if (parameterNames != null && parameterNames.length > 0) {
for (String parameterName : parameterNames) {
params.put(parameterName, joinPoint.getArgs()[i]);
i++;
}
}
return params;
}
public static boolean atLeastOnePropertyNotNull(Object obj) {
for (Field field : obj.getClass().getDeclaredFields()) {
//忽略serialVersionUID
if ("serialVersionUID".equals(field.getName())) {
continue;
}
field.setAccessible(true);
try {
if (field.get(obj) != null && !field.get(obj).toString().isEmpty()) {
return true;
}
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
return false;
}
/**
* 方法后调用
*/
@After("serviceAspect()")
public void doAfterInService(JoinPoint joinPoint) {
}
}
然后,解释一下
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
这是我们的第一行代码,这里的joinPoint对象表示应用建议的程序执行点,getSignature()方法则会返回正在执行的方法的方法签名,签名里就包含了该方法名称、返回类型和参数类型。然后再强制转换成MethodSignature
对象,便于访问方法相关的信息。
MethodSignature是一个对象,它表示正在执行的方法的签名,包括方法名称、返回类型和参数类型。它是Spring AOP框架中的一个类,用于在切面中获取方法的信息。
Method method = signature.getMethod();获取方法
这个则是从MethodSignature中获取到了正在执行的方法信息。
LocalVariableTableParameterNameDiscoverer u = new LocalVariableTableParameterNameDiscoverer();
LocalVariableTableParameterNameDiscoverer是一个类,它可以用于在运行时获取方法参数的名称。它是Spring框架中的一个类,通常与反射一起使用,以便在运行时获取有关方法参数的信息。
通过 String[] parameterNames = u.getParameterNames(method);
可以获取到目前方法的入参名称。
getParamMap();
方法则会返回一个key是参数名称,value是参数本身的map对象。 我们可以从中取出我们需要的那个参数对象。
VerifyParameters verifyParameters = method.getAnnotation(VerifyParameters.class);
上边这行代码则会获取到方法上注解的注解对象本身,和我们传入的参数值,因为每个方法的入参不尽相同,里边时间的字段也不尽相同,需要主动传入来做处理。
这里再解释下atLeastOnePropertyNotNull()
方法,这个方法的作用是判断参数内的属性是否至少有一个不为空,这里我们忽略了serialVersionUID
。
serialVersionUID是Java中的一个序列化机制,用于在反序列化期间验证发送方和接收方的对象是否具有兼容的序列化版本。如果发送方和接收方的serialVersionUID不同,则反序列化将失败。如果未指定serialVersionUID,则Java运行时将根据类的特定方面自动生成它。
之后的方法就很简单了,拿出值根据我们的需要做处理即可
controller
再看看是如何使用的吧,添加@VerifyParameters
注解,传入时间的属性名称,和需要校验的参数名称,这里传入参数名称的原因是,可能和我这里一样还有其他的参数影响,而我们只想校验我们需要的参数。
@ApiOperation(value = "查询列表")
@GetMapping("/test")
@VerifyParameters(startTimeParamName = "beginTime", endTimeParamName = "endTime",paramName = "orderRequest")
public Page<Map<String, Object>> findOrderTestList(Pageable pageable, ERPOrderRequest orderRequest) {
log.info("模拟逻辑处理");
return null;
}
这样我们就实现了通过一个自定义注解对方法的入参进行了校验,在取到入参和方法的各个值之后,我们其实可以做各种各样的操作,各位小伙伴可以任意发挥。
来源:https://juejin.cn/post/7221131892426571837


猜你喜欢
- MainActivity如下:package cc.ab;import android.os.Bundle;import android.p
- springboot整合tkmapper1.导入pom依赖1.1 导入springboot的parent依赖<parent> &
- Linux下的五种I/O模型1)阻塞I/O(blocking I/O)2)非阻塞I/O (nonblocking I/O)3) I/O复用(
- 我这一次讲使用scroll实现弹性滑动,我不会只有一个例子就说完,因为写文章的时候我也在学习,我分几次讲完吧。首先上一段代码,private
- 该方法针对idea版本(2020.2.x)C:\Users\yanghao\AppData\Roaming\JetBrains\Intell
- 本文实例讲述了Java 线程的生命周期。分享给大家供大家参考,具体如下:一 代码/*** @Title: ThreadStatus.java
- 1.java创建自定义类数组方法:Student []stu = new Student[3];for(int i = 0; i <
- 数学工具类Math,供大家参考,具体内容如下1. 概述java.util.Math类是数学相关的工具类,里面提供了大量的静态方法,完成与数学
- ArrrayList是Java中经常被用到的集合,弄清楚它的底层实现,有利于我们更好地使用它。下图是ArrayList的UML图从图中我们可
- 目录1、Android如何动态更换桌面图标1.1使用场景1.2知识点1.3使用Activity-alias2、巨坑2.1App的覆盖2.2桌
- private void Value_ByteArray(){double doublevalue = 258.0;Int32 intval
- 一、项目简述功能:用户的邮箱注册、验证码验证以及用户登录。 不需要注册账号,也可以上传满足条件的临时文件,但是只4小时内有效。 文件的管理,
- SpringMVC异常处理机制(一)项目前准备首先参照文章Spring课程工程构建+SpringMVC简介及其快速入门搭建项目搭建好一个项目
- 本文实例讲述了Java实现分解任意输入数的质因数算法。分享给大家供大家参考,具体如下:分解任意输入数的质因数:质因数概念:任何一个合数都可以
- 基本概念 * (Listener): * 用于监听web应用中某些对象、信息的创建、销毁、增加,修改,删除等动作的发生,然后作出相应的响应处
- 一、简介约瑟夫问题(有时也称为约瑟夫斯置换,是一个出现在计算机科学和数学中的问题。在计算机编程的算法中,类似问题又称为约瑟夫环。又称“丢手绢
- 背景现行的文本编辑器大多都具备文本查询的能力,但是并不能直观的告诉用户两段文字的细微差异,所以对比工具在某种情况下,就起到了很便捷的效率。关
- struct InputStream 是单个输入流的管理器。是由 add_input_stream() 函数申
- 一、准备工作mybatis-plus作为mybatis的增强工具,它的出现极大的简化了开发中的数据库操作,但是长久以来,它的联表查询能力一直
- Android 关闭多个Activity的实现方法总的来说有四种方法:1、使用Application来进行关闭public class Ap