自定义防止重复提交注解@PreventDuplicateSubmit,Spring Aop

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD, ElementType.TYPE})
public @interface PreventDuplicateSubmit {
    /** 请求间隔时间,单位秒,该时间范围内的请求为重复请求 */
    int intervalTime() default 1;
    /** 返回的提示信息 */
    String msg() default "请不要频繁重复请求!";
}
@Component
public class PreventDuplicateSubmitCache {

    /**
     * 重复提交缓存
     */
    public static final Map<Date, String> REPEAT_REQUEST = new HashMap<>();

    /**
     * 向缓存中存入数据
     *
     * @param key
     * @param value
     */
    public void add(Date key, String value) {
        REPEAT_REQUEST.put(key, value);
    }

}
@Component
@Aspect
@Order(100)
public class PreventDuplicateSubmitHandler {
    @Resource
    private PreventDuplicateSubmitCache requestCache;

    // 定义注解类型的切点
    @Pointcut("@annotation(fish.api.server.security.PreventDuplicateSubmit)")
    public void arrPointcut() {
    }

    // 实现过滤重复请求功能
    @Around("arrPointcut()")
    public Object arrBusiness(ProceedingJoinPoint joinPoint) {

        // 获取 map value,由 session id 和 请求URI 构成
        ServletRequestAttributes sra = (ServletRequestAttributes) RequestContextHolder.currentRequestAttributes();
        HttpServletRequest request = sra.getRequest();
        String value = request.getSession().getId() + "_" + request.getRequestURI();

        // 获取方法的 AvoidRepeatRequest 注解
        Method method = ((MethodSignature) joinPoint.getSignature()).getMethod();
        PreventDuplicateSubmit arr = method.getAnnotation(PreventDuplicateSubmit.class);

        // 判断是否是重复的请求
        Date now = new Date();
        requestCache.add(now, value);
        List<Date> list = new ArrayList<>();
        Map<Date, String> map = PreventDuplicateSubmitCache.REPEAT_REQUEST;
        for (Map.Entry<Date, String> entry : map.entrySet()) {
            if (entry.getKey().equals(now) && entry.getValue().equals(value)) {
                continue;
            }
            list.add(entry.getKey());
            //传入的时间和原本时间的差值(秒)
            int difference = (int) (now.getTime() - entry.getKey().getTime()) / 1000;
            //如果时间差小于设定值为重复提交
            if (difference < arr.intervalTime()) {
                return new Result<>(902, arr.msg());
            }
        }
        //删除之前的数据
        if (CollUtil.isNotEmpty(list)) {
            list.forEach(map::remove);
        }


        try {
            // 非重复请求,执行业务代码
            return joinPoint.proceed();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
            return ("error");
        }
    }

}

应用

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
自定义注解AOP重复提交是一种通过在代码中添加自定义注解的方式来实现防止重复提交的功能。这种方法可以有效地避免代码耦合性过强,提高代码的可读性和可维护性。 具体实现的方案可以有多种,以下是几种常见的方案: 1. Token验证:在每次请求中添加一个唯一的Token标识,服务端接收到请求后将Token保存在缓存中,然后进行重复提交的验证。如果同一个Token已经存在于缓存中,则表示该请求已经被处理过,可以拒绝重复提交。 2. 请求参数验证:通过对请求参数进行校验,判断是否已经存在相同的请求参数,如果存在则表示重复提交。可以使用缓存或者数据库来存储已经处理过的请求参数,通过查询来进行重复提交的验证。 3. 时间窗口验证:通过设置一个时间窗口,限制在该时间窗口内只接受一次请求。可以使用缓存或者数据库记录请求的时间戳,每次接收到请求时与最近一次的时间戳进行比对,如果在时间窗口内已经存在过请求,则拒绝重复提交。 以上方案都可以使用Redis作为缓存来进行存储和验证操作。可以通过引入相关的依赖来使用Spring Boot集成的Redis组件和Jedis依赖。 通过使用自定义注解AOP来实现防重复提交可以有效地提高代码的可读性和可维护性,同时也能够减轻服务器的负载,避免因为重复提交而导致的服务器宕机等问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值