实现逻辑
1.自定义防重复提交的注解和切面
2.在需要验证的接口上增加注解(一般是创建、修改的接口)
3.以每次调用的 用户唯一标识(userId或者sessionId或者token)+ 请求路径+参数 作为key,value任意值都可以,缓存起来(redis或本地缓存),并设置一个合适的缓存失效时间。
4.每次调用时根据key判断,缓存是否存在,存在则抛出异常或提示,不存在则执行业务逻辑
代码
防重复提交注解
/**
* @功能描述 防止重复提交标记注解
* @author gourd
* @date 2018-08-26
*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface NoRepeatSubmit {
/**
* 重复统计时长
* @return
*/
int time() default 1;
}
/**
* 接口防重复切面
* @author gourd
* @date 2018-08-26
*/
@Aspect
@Component
@Slf4j
public class NoRepeatSubmitAop {
private static final String JWT_TOKEN_KEY = "jwt-token";
@Pointcut("@annotation(com.gourd.hu.base.annotation.NoRepeatSubmit)")
public void serviceNoRepeat() {
}
@Around("serviceNoRepeat()")
public Object around(ProceedingJoinPoint pjp) throws Throwable {
HttpServletRequest request = RequestHolder.getRequest();
String jwtToken = request.getHeader(JWT_TOKEN_KEY);
String key = Md5Util.md5(jwtToken + "-" + request.getRequestURL().toString()+"-"+ JSON.toJSONString(request.getParameterMap()));
if (RedisUtil.get(key) == null) {
try {
Object o = pjp.proceed();
MethodSignature signature = (MethodSignature) pjp.getSignature();
NoRepeatSubmit noRepeatSubmit = signature.getMethod().getAnnotation(NoRepeatSubmit.class);
// 默认1秒内统一用户同一个地址同一个参数,视为重复提交
RedisUtil.setExpire(key, "0",noRepeatSubmit.time());
return o;
}catch (Exception e){
throw new ServiceException(e.getMessage(),e.getCause());
}
} else {
throw new ServiceException("重复提交");
}
}
}
@PostMapping("/repeatSubmit")
@NoRepeatSubmit(time = 2)
@ApiOperation(value="测试重复提交")
public String testRepeatSubmit() {
return ("测试重复提交程序逻辑返回");
}
至此就完成了
————————————————
版权声明:本文为CSDN博主「葫芦胡」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/HXNLYW/java/article/details/94560771
网友评论