接口防抖处理

最近在做项目中遇到一个接口由于用户“手抖”向库里插入了多条id不同其他字段相同的数据,未解决这个问题,查阅资料,整理此笔记,方便下次查看。

接口防抖处理是防止短时间内多次触发相同请求的一种技术手段,通常用于避免重复提交表单、重复点击按钮等场景。

在实现接口防抖可以通过以下几种方式:

1. 基于时间戳的防抖

通过记录请求的时间戳,判断两次请求的时间间隔是否小于某个阈值(如5秒)。如果小于阈值,则认为是重复请求并拒绝处理。

@PostMapping("/enterTheWarehouse")
public AjaxResult enterTheWarehouse(@Validated @RequestBody InWarehouseVO inWarehouseVO) {
    // 使用请求参数生成唯一的 key
    String key = "/enterTheWarehouse-" + inWarehouseVO.getAccountId();

    // 获取当前时间戳
    long currentTimestamp = System.currentTimeMillis();
    log.info("当前时间戳:{}", currentTimestamp);

    // 上一次请求的时间戳
    Long lastTimestamp = requestTimestamps.get(key);
    log.info("上一次请求时间戳:{}", lastTimestamp);

    // 如果上一次请求时间不为空,并且与当前时间间隔小于5000毫秒(5秒),则认为是重复请求
    if (lastTimestamp != null && currentTimestamp - lastTimestamp < 5000) {
        return AjaxResult.error("重复提交,请稍后再试!");
    }

    // 记录当前请求时间戳
    requestTimestamps.put(key, currentTimestamp);

    this.storageToolService.enterTheWarehouse(inWarehouseVO);
    return AjaxResult.success();
}

2. 基于令牌的防抖

在用户发起请求时,生成一个唯一的令牌(Token),并将该令牌存储在服务器端或客户端。每次请求时都需要携带这个令牌,服务器端通过验证令牌来判断是否是重复请求。

实现思路

  1. 生成令牌:在用户发起请求时,生成一个唯一的令牌(Token)。
  2. 存储令牌:将生成的令牌存储在服务器端(如内存、数据库或缓存)。
  3. 验证令牌:每次请求时,检查令牌
Spring Boot 中的接口防抖(Throttle or Debounce)通常是指为了限制频繁调用某个接口而采取的一种策略,它会在用户连续多次请求之间设置一段延迟,只有当这个时间间隔超过预设值后,才会允许再次发送请求。这在处理高并发、防止滥用或服务降级时很有用。 Spring Boot 并没有直接提供这样的功能,但是可以结合第三方库如 Spring Cloud Gateway 或者自定义过滤器(Filter)来实现。你可以利用滑动窗口算法(Sliding Window Algorithm)或者令牌桶算法(Token Bucket)来控制接口访问频率。例如,可以使用 `RateLimiter` 或者 Redis 的过期键来管理访问令牌。 以下是简单的一个示例,使用 Spring Cloud Gateway 的 Filter: ```java import org.springframework.cloud.gateway.filter.ratelimit.KeyResolver; import reactor.core.publisher.Mono; public class CustomKeyResolver implements KeyResolver { private final RateLimiter rateLimiter; public CustomKeyResolver(RateLimiter rateLimiter) { this.rateLimiter = rateLimiter; } @Override public Mono<String> resolve(Object exchange) { // 这里可以根据实际情况提取请求关键信息作为 key String apiKey = extractApiKeyFromExchange(exchange); return Mono.just(apiKey).doOnNext(key -> { if (!rateLimiter.tryAcquire()) { // 如果没有获取到令牌,则记录防抖 log.info("Debouncing API call for key {}", apiKey); } }); } private String extractApiKeyFromExchange(Object exchange) { // 实现从exchange中提取API密钥的逻辑 } } // 在GlobalFilter配置中启用 @Bean public GlobalFilter customRateLimitingFilter(GatewayFilterChain chain, RateLimiter rateLimiter) { return new ConditionalGlobalFilter( (exchange) -> true, // 条件总是true,表示始终开启限流 new CustomKeyResolver(rateLimiter), chain ); } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值