SpringBoot利用限速器RateLimiter實現(xiàn)單機限流

一. 概述

參考開源項目https://github.com/xkcoding/spring-boot-demo
在系統(tǒng)運維中, 有時候為了避免用戶的惡意刷接口, 會加入一定規(guī)則的限流, 本Demo使用速率限制器com.xkcoding.ratelimit.guava.annotation.RateLimiter實現(xiàn)單機版的限流

二. SpringBootDemo

2.1 依賴

    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-web</artifactId>
    </dependency>

    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-aop</artifactId>
    </dependency>

    <dependency>
      <groupId>cn.hutool</groupId>
      <artifactId>hutool-all</artifactId>
    </dependency>

    <dependency>
      <groupId>com.google.guava</groupId>
      <artifactId>guava</artifactId>
    </dependency>

2.2 application.yml

server:
  port: 8080
  servlet:
    context-path: /demo

2.3 啟動類

@SpringBootApplication
public class SpringBootDemoRatelimitGuavaApplication {
    public static void main(String[] args) {
        SpringApplication.run(SpringBootDemoRatelimitGuavaApplication.class, args);
    }
}

2.4 定義一個限流注解 RateLimiter.java

注意代碼里使用了 AliasFor 設(shè)置一組屬性的別名布近,所以獲取注解的時候堡赔,需要通過 Spring 提供的注解工具類 AnnotationUtils 獲取羽圃,不可以通過 AOP 參數(shù)注入的方式獲取牛曹,否則有些屬性的值將會設(shè)置不進去。

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RateLimiter {
    int NOT_LIMITED = 0;

    /**
     * qps (每秒并發(fā)量)
     */
    @AliasFor("qps") double value() default NOT_LIMITED;

    /**
     * qps (每秒并發(fā)量)
     */
    @AliasFor("value") double qps() default NOT_LIMITED;

    /**
     * 超時時長,默認不等待
     */
    int timeout() default 0;

    /**
     * 超時時間單位,默認毫秒
     */
    TimeUnit timeUnit() default TimeUnit.MICROSECONDS;
}

2.5 代理: RateLimiterAspect.java

@Slf4j
@Aspect
@Component
public class RateLimiterAspect {
    /**
     * 單機緩存
     */
    private static final ConcurrentMap<String, com.google.common.util.concurrent.RateLimiter> RATE_LIMITER_CACHE = new ConcurrentHashMap<>();

    /**
      這里記得改成你自己注解實際的包路徑,我看有些小伙伴說報錯, 就是路徑不對識別不到
    */
    @Pointcut("@annotation(com.**.RateLimiter)")
    public void rateLimit() {

    }

    @Around("rateLimit()")
    public Object pointcut(ProceedingJoinPoint point) throws Throwable {
        MethodSignature signature = (MethodSignature) point.getSignature();
        Method method = signature.getMethod();
        // 通過 AnnotationUtils.findAnnotation 獲取 RateLimiter 注解
        RateLimiter rateLimiter = AnnotationUtils.findAnnotation(method, RateLimiter.class);
        if (rateLimiter != null && rateLimiter.qps() > RateLimiter.NOT_LIMITED) {
            double qps = rateLimiter.qps();
            // TODO 這個key可以根據(jù)具體需求配置,例如根據(jù)ip限制,或用戶
            String key = method.getDeclaringClass().getName() + StrUtil.DOT + method.getName();
            if (RATE_LIMITER_CACHE.get(key) == null) {
                // 初始化 QPS
                RATE_LIMITER_CACHE.put(key, com.google.common.util.concurrent.RateLimiter.create(qps));
            }

            // 嘗試獲取令牌
            if (RATE_LIMITER_CACHE.get(key) != null && !RATE_LIMITER_CACHE.get(key).tryAcquire(rateLimiter.timeout(), rateLimiter.timeUnit())) {
                throw new RuntimeException("手速太快了瓢娜,慢點兒吧~");
            }
        }
        return point.proceed();
    }
}

2.6 使用

@Slf4j
@RestController
public class TestController {

    /**
     * 接口每秒只能請求一次,不等待
     * @return
     */
    @RateLimiter(value = 1.0)
    @GetMapping("/test1")
    public Dict test1() {
        log.info("【test1】被執(zhí)行了挂洛。。眠砾。虏劲。。");
        return Dict.create().set("msg", "hello,world!").set("description", "別想一直看到我荠藤,不信你快速刷新看看~");
    }

    /**
     * 接口每秒只能請求一次,等待一秒
     * @return
     */
    @RateLimiter(value = 1.0, timeout = 1,timeUnit = TimeUnit.SECONDS)
    @GetMapping("/test3")
    public Dict test3() {
        log.info("【test3】被執(zhí)行了伙单。。哈肖。吻育。。");
        return Dict.create().set("msg", "hello,world!").set("description", "別想一直看到我淤井,不信你快速刷新看看~");
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末布疼,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子币狠,更是在濱河造成了極大的恐慌游两,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件漩绵,死亡現(xiàn)場離奇詭異贱案,居然都是意外死亡,警方通過查閱死者的電腦和手機止吐,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進店門宝踪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人碍扔,你說我怎么就攤上這事瘩燥。” “怎么了不同?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵厉膀,是天一觀的道長溶耘。 經(jīng)常有香客問我,道長服鹅,這世上最難降的妖魔是什么凳兵? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮菱魔,結(jié)果婚禮上留荔,老公的妹妹穿的比我還像新娘吟孙。我一直安慰自己澜倦,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布杰妓。 她就那樣靜靜地躺著藻治,像睡著了一般。 火紅的嫁衣襯著肌膚如雪巷挥。 梳的紋絲不亂的頭發(fā)上桩卵,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天,我揣著相機與錄音倍宾,去河邊找鬼雏节。 笑死,一個胖子當(dāng)著我的面吹牛高职,可吹牛的內(nèi)容都是我干的钩乍。 我是一名探鬼主播,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼怔锌,長吁一口氣:“原來是場噩夢啊……” “哼寥粹!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起埃元,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤涝涤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后岛杀,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體阔拳,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年类嗤,在試婚紗的時候發(fā)現(xiàn)自己被綠了糊肠。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡土浸,死狀恐怖罪针,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情黄伊,我是刑警寧澤泪酱,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響墓阀,放射性物質(zhì)發(fā)生泄漏毡惜。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一斯撮、第九天 我趴在偏房一處隱蔽的房頂上張望经伙。 院中可真熱鬧,春花似錦勿锅、人聲如沸帕膜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽垮刹。三九已至,卻和暖如春张弛,著一層夾襖步出監(jiān)牢的瞬間荒典,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工吞鸭, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留寺董,地道東北人。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓刻剥,卻偏偏與公主長得像遮咖,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子透敌,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,722評論 2 345

推薦閱讀更多精彩內(nèi)容