亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java通過(guò)Caffeine和自定義注解實(shí)現(xiàn)本地防抖接口限流

 更新時(shí)間:2025年05月20日 08:32:33   作者:不掉頭發(fā)的阿水  
Caffeine?是目前Java領(lǐng)域最熱門(mén),性能最高的本地內(nèi)存緩存庫(kù),本文為大家詳細(xì)介紹了如何利用Caffeine和自定義注解實(shí)現(xiàn)本地防抖接口限流功能,需要的可以了解下

一、背景與需求

在實(shí)際項(xiàng)目開(kāi)發(fā)中,經(jīng)常遇到接口被前端高頻觸發(fā)、按鈕被多次點(diǎn)擊或者接口重復(fù)提交的問(wèn)題,導(dǎo)致服務(wù)壓力變大、數(shù)據(jù)冗余、甚至引發(fā)冪等性/安全風(fēng)險(xiǎn)。

常規(guī)做法是前端節(jié)流/防抖、后端用Redis全局限流、或者API網(wǎng)關(guān)限流。但在很多場(chǎng)景下:

  • 接口只要求單機(jī)(本地)防抖,不需要全局一致性;
  • 只想讓同一個(gè)業(yè)務(wù)對(duì)象(同一手機(jī)號(hào)、同一業(yè)務(wù)ID、唯一標(biāo)識(shí))在自定義設(shè)置秒內(nèi)只處理一次;
  • 想要注解式配置,讓代碼更優(yōu)雅、好維護(hù)。

這個(gè)時(shí)候,Caffeine+自定義注解+AOP的本地限流(防抖)方案非常合適。

二、方案設(shè)計(jì)

1. Caffeine介紹

Caffeine 是目前Java領(lǐng)域最熱門(mén)、性能最高的本地內(nèi)存緩存庫(kù),QPS可達(dá)百萬(wàn)級(jí),適用于低延遲、高并發(fā)、短TTL緩存場(chǎng)景。
在本地限流、防抖、接口去重等方面天然有優(yōu)勢(shì)。

2. 自定義注解+AOP

用自定義注解(如@DebounceLimit)標(biāo)記要防抖的接口,AOP切面攔截后判斷是否需要限流,核心思路是:

  • 以唯一標(biāo)識(shí)作為key;
  • 每次訪問(wèn)接口,先查詢本地Caffeine緩存;
  • 如果key在2秒內(nèi)已被處理過(guò),則直接攔截;
  • 否則執(zhí)行業(yè)務(wù)邏輯,并記錄處理時(shí)間。

這種方式無(wú)侵入、代碼簡(jiǎn)潔、可擴(kuò)展性強(qiáng),適合絕大多數(shù)本地場(chǎng)景。

效果圖如下:

三、完整實(shí)現(xiàn)步驟

1.Pom依賴如下

        <dependency>
            <groupId>com.github.ben-manes.caffeine</groupId>
            <artifactId>caffeine</artifactId>
            <version>2.9.3</version>
        </dependency>
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
        </dependency>

2. 定義自定義注解

import java.lang.annotation.*;
 
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface DebounceLimit {
    /**
     * 唯一key(支持SpEL表達(dá)式,如 #dto.id)
     */
    String key();
 
    /**
     * 防抖時(shí)間,單位秒
     */
    int ttl() default 2;
 
    /**
     * 是否返回上次緩存的返回值
     */
    boolean returnLastResult() default true;
}

3. 配置Caffeine緩存Bean

import com.github.benmanes.caffeine.cache.Cache;
import com.github.benmanes.caffeine.cache.Caffeine;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
 
import java.util.concurrent.TimeUnit;
 
@Configuration
public class DebounceCacheConfig {
    @Bean
    public Cache<String, Object> debounceCache() {
        return Caffeine.newBuilder()
                .expireAfterWrite(1, TimeUnit.MINUTES)
                .maximumSize(100_000)
                .build();
    }
}

4. 編寫(xiě)AOP切面

import com.github.benmanes.caffeine.cache.Cache;
import com.lps.anno.DebounceLimit;
import lombok.Getter;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.expression.ExpressionParser;
import org.springframework.expression.spel.standard.SpelExpressionParser;
import org.springframework.expression.spel.support.StandardEvaluationContext;
import org.springframework.stereotype.Component;
import java.lang.reflect.Method;
 
@Slf4j
@Aspect
@Component
public class DebounceLimitAspect {
 
    @Autowired
    private Cache<String, Object> debounceCache;
 
    private final ExpressionParser parser = new SpelExpressionParser();
 
    @Around("@annotation(debounceLimit)")
    public Object around(ProceedingJoinPoint pjp, DebounceLimit debounceLimit) throws Throwable {
        // 1. 獲取方法、參數(shù)
        MethodSignature methodSignature = (MethodSignature) pjp.getSignature();
        Method method = methodSignature.getMethod();
        Object[] args = pjp.getArgs();
        String[] paramNames = methodSignature.getParameterNames();
        StandardEvaluationContext context = new StandardEvaluationContext();
        for (int i = 0; i < paramNames.length; i++) {
            context.setVariable(paramNames[i], args[i]);
        }
 
        // 2. 解析SpEL表達(dá)式得到唯一key
        String key = parser.parseExpression(debounceLimit.key()).getValue(context, String.class);
        String cacheKey = method.getDeclaringClass().getName() + "." + method.getName() + ":" + key;
 
        long now = System.currentTimeMillis();
        DebounceResult<Object> debounceResult = (DebounceResult<Object>) debounceCache.getIfPresent(cacheKey);
 
        if (debounceResult != null && (now - debounceResult.getTimestamp() < debounceLimit.ttl() * 1000L)) {
            String methodName = pjp.getSignature().toShortString();
            log.error("接口[{}]被限流, key={}", methodName, cacheKey);
            // 是否返回上次結(jié)果
            if (debounceLimit.returnLastResult() && debounceResult.getResult() != null) {
                return debounceResult.getResult();
            }
            // 統(tǒng)一失敗響應(yīng),可自定義異常或返回結(jié)構(gòu)
            return new RuntimeException("操作過(guò)于頻繁,請(qǐng)稍后再試!");
        }
 
        Object result = pjp.proceed();
        debounceCache.put(cacheKey, new DebounceResult<>(result, now));
        return result;
    }
 
    @Getter
    static class DebounceResult<T> {
        private final T result;
        private final long timestamp;
 
        public DebounceResult(T result, long timestamp) {
            this.result = result;
            this.timestamp = timestamp;
        }
    }
}

5. 控制器里直接用注解實(shí)現(xiàn)防抖

@RestController
@RequiredArgsConstructor
@Slf4j
public class DebounceControl {
    private final UserService userService;
    @PostMapping("/getUsernameById")
    @DebounceLimit(key = "#dto.id", ttl = 10)
    public String test(@RequestBody User dto) {
        log.info("在{}收到了請(qǐng)求,參數(shù)為:{}", DateUtil.now(), dto);
        return userService.getById(dto.getId()).getUsername();
    }
}

只要加了這個(gè)注解,同一個(gè)id的請(qǐng)求在自定義設(shè)置的秒內(nèi)只處理一次,其他直接被攔截并打印日志。

四、擴(kuò)展與注意事項(xiàng)

1.SpEL表達(dá)式靈活

可以用 #dto.id、#dto.mobile、#paramName等,非常適合多參數(shù)、復(fù)雜唯一性業(yè)務(wù)場(chǎng)景。

2.returnLastResult適合有“緩存返回結(jié)果”的場(chǎng)景

比如查詢接口、表單重復(fù)提交直接復(fù)用上次的返回值。

3.本地限流僅適用于單機(jī)環(huán)境

多節(jié)點(diǎn)部署建議用Redis分布式限流,原理一樣。

4.緩存key建議加上方法簽名

避免不同接口之間key沖突。

5.Caffeine最大緩存、過(guò)期時(shí)間應(yīng)根據(jù)業(yè)務(wù)并發(fā)和內(nèi)存合理設(shè)置

絕大多數(shù)接口幾千到幾萬(wàn)key都沒(méi)壓力。

五、適用與不適用場(chǎng)景

適用:

  • 單機(jī)接口防抖/限流
  • 短時(shí)間重復(fù)提交防控
  • 按業(yè)務(wù)唯一標(biāo)識(shí)維度防刷
  • 秒殺、報(bào)名、投票等接口本地保護(hù)

不適用:

  • 分布式場(chǎng)景(建議用Redis或API網(wǎng)關(guān)限流)
  • 需要全局一致性的業(yè)務(wù)
  • 內(nèi)存非常敏感/極端高并發(fā)下,需結(jié)合Redis做混合限流

六、總結(jié)

Caffeine + 注解 + AOP的本地限流防抖方案,實(shí)現(xiàn)簡(jiǎn)單、代碼優(yōu)雅、性能極高、擴(kuò)展靈活

到此這篇關(guān)于Java通過(guò)Caffeine和自定義注解實(shí)現(xiàn)本地防抖接口限流的文章就介紹到這了,更多相關(guān)Java本地防抖接口限流內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java垃圾回收機(jī)制的示例詳解

    Java垃圾回收機(jī)制的示例詳解

    本文主要圍繞著Java垃圾回收當(dāng)中的哪些內(nèi)存需要回收?什么時(shí)候回收?如何回收?進(jìn)行了詳細(xì)講解,感興趣的小伙伴可以學(xué)習(xí)一下
    2022-04-04
  • JAVA?兩個(gè)類同時(shí)實(shí)現(xiàn)同一個(gè)接口的方法(三種方法)

    JAVA?兩個(gè)類同時(shí)實(shí)現(xiàn)同一個(gè)接口的方法(三種方法)

    在Java中,兩個(gè)類同時(shí)實(shí)現(xiàn)同一個(gè)接口是非常常見(jiàn)的,接口定義了一組方法,實(shí)現(xiàn)接口的類必須提供這些方法的具體實(shí)現(xiàn),以下將展示如何實(shí)現(xiàn)這一要求,并提供具體的代碼示例,需要的朋友可以參考下
    2024-08-08
  • 23種設(shè)計(jì)模式(9) java橋接模式

    23種設(shè)計(jì)模式(9) java橋接模式

    這篇文章主要為大家詳細(xì)介紹了java設(shè)計(jì)模式之橋接模式,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-11-11
  • 關(guān)于SpringBoot中Ajax跨域以及Cookie無(wú)法獲取丟失問(wèn)題

    關(guān)于SpringBoot中Ajax跨域以及Cookie無(wú)法獲取丟失問(wèn)題

    這篇文章主要介紹了關(guān)于SpringBoot中Ajax跨域以及Cookie無(wú)法獲取丟失問(wèn)題,本文具有參考意義,遇到相同或者類似問(wèn)題的小伙伴希望可以從中找到靈感
    2023-03-03
  • Spring?Security?OAuth?Client配置加載源碼解析

    Spring?Security?OAuth?Client配置加載源碼解析

    這篇文章主要為大家介紹了Spring?Security?OAuth?Client配置加載源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-07-07
  • Java中ArrayList集合的常用方法大全

    Java中ArrayList集合的常用方法大全

    這篇文章主要給大家介紹了關(guān)于Java中ArrayList集合的常用方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • java網(wǎng)上圖書(shū)商城(2)Category模塊

    java網(wǎng)上圖書(shū)商城(2)Category模塊

    這篇文章主要介紹了java網(wǎng)上圖書(shū)商城,Category模塊,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-12-12
  • 詳解mybatis插入數(shù)據(jù)后返回自增主鍵ID的問(wèn)題

    詳解mybatis插入數(shù)據(jù)后返回自增主鍵ID的問(wèn)題

    這篇文章主要介紹了mybatis插入數(shù)據(jù)后返回自增主鍵ID詳解,本文通過(guò)場(chǎng)景分析示例代碼相結(jié)合給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2021-07-07
  • springBoot項(xiàng)目打包idea的多種方法

    springBoot項(xiàng)目打包idea的多種方法

    這篇文章主要介紹了springBoot項(xiàng)目打包idea的多種方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-07-07
  • Mybatis結(jié)果集映射一對(duì)多簡(jiǎn)單入門(mén)教程

    Mybatis結(jié)果集映射一對(duì)多簡(jiǎn)單入門(mén)教程

    本文給大家介紹Mybatis結(jié)果集映射一對(duì)多簡(jiǎn)單入門(mén)教程,包括搭建數(shù)據(jù)庫(kù)環(huán)境的過(guò)程,idea搭建maven項(xiàng)目的代碼詳解,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧
    2021-06-06

最新評(píng)論