需求说明
在项目开发过程,我们也会经常遇到这种问题,前端未拦截,或者拦截失败,导致后端接收到大量重复请求,结果把这些重复请求入库后,产生大量垃圾数据。
解释下幂等的概念:
任意多次执行所产生的影响均与一次执行的影响相同
那这个听起来是问题的解决之道,访如何实现,通常有如下方式:
- 建立数据库索引,保证最终插入的只有一条数据;
- token机制,每次接口请求前先获取一个token,然后再下次请求的时候在请求的header体中加上这个token,后台进行验证,如果验证通过删除token,下次请求再次判断token;
- 悲观锁或者乐观锁,悲观锁可以保证每次for update的时候其他sql无法update数据(在数据库引擎是innodb的时候,select的条件必须是唯一索引,防止锁全表)
- 先查询后判断,首先通过查询数据库是否存在数据,如果存在证明已经请求过了,直接拒绝该请求,如果没有存在,就证明是第一次进来,直接放行。
幂等的原理图
实战演练
实战中对上述进行一些优化,支持Header和判断参数值两种形式,详细看如下代码:
package vip.mate.core.ide.enums;
import lombok.AllArgsConstructor;
import lombok.Getter;
/**
* 幂等枚举类
*
* @author pangu
*/
@Getter
@AllArgsConstructor
public enum IdeTypeEnum {
/**
* 0+1
*/
ALL(0, "ALL"),
/**
* ruid 是针对每一次请求的
*/
RID(1, "RID"),
/**
* key+val 是针对相同参数请求
*/
KEY(2, "KEY");
private final Integer index;
private final String title;
}
拦截器业务逻辑
package vip.mate.core.ide.aspect;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import io.micrometer.core.instrument.util.StringUtils;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.CodeSignature;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import vip.mate.core.common.aspect.BaseAspect;
import vip.mate.core.ide.annotation.Ide;
import vip.mate.core.ide.enums.IdeTypeEnum;
import vip.mate.core.