精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Java實(shí)現(xiàn)系統(tǒng)限流

開發(fā)
本文主要介紹了幾種限流方法:Guava RateLimiter、簡(jiǎn)單計(jì)數(shù)、滑窗計(jì)數(shù)、信號(hào)量、令牌桶。

限流是保障系統(tǒng)高可用的方式之一,也是大廠高頻面試題,如果面試官問一句,“如何實(shí)現(xiàn)每秒鐘1000個(gè)請(qǐng)求的限流?”,你要是分分鐘給他寫上幾種限流方案,那豈不香哉,哈哈!話不多說,我來列幾種常用限流實(shí)現(xiàn)方式。

1、Guava RateLimiter

Guava是Java領(lǐng)域很優(yōu)秀的開源項(xiàng)目,包含了日常開發(fā)常用的集合、String、緩存等, 其中RateLimiter是常用限流工具。

RateLimiter是基于令牌桶算法實(shí)現(xiàn)的,如果每秒10個(gè)令牌,內(nèi)部實(shí)現(xiàn),會(huì)每100ms生產(chǎn)1個(gè)令牌。

使用Guava RateLimiter,如下:

(1) 引入pom依賴:

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

(2) 代碼:

public class GuavaRateLimiterTest {
    //比如每秒生產(chǎn)10個(gè)令牌,相當(dāng)于每100ms生產(chǎn)1個(gè)令牌
    private RateLimiter rateLimiter = RateLimiter.create(10);

    /**
     * 模擬執(zhí)行業(yè)務(wù)方法
     */
    public void exeBiz() {
        if (rateLimiter.tryAcquire(1)) {
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("線程" + Thread.currentThread().getName() + ":執(zhí)行業(yè)務(wù)邏輯");
        } else {
            System.out.println("線程" + Thread.currentThread().getName() + ":被限流");
        }
    }

    public static void main(String[] args) throws InterruptedException {
        GuavaRateLimiterTest limiterTest = new GuavaRateLimiterTest();
        Thread.sleep(500);//等待500ms,讓limiter生產(chǎn)一些令牌

        //模擬瞬間生產(chǎn)100個(gè)線程請(qǐng)求
        for (int i = 0; i < 100; i++) {
            new Thread(limiterTest::exeBiz).start();
        }
    }
}

2、滑窗計(jì)數(shù)

打個(gè)比方,某接口每秒允許100個(gè)請(qǐng)求,設(shè)置一個(gè)滑窗,窗口中有10個(gè)格子,每個(gè)格子占100ms,每100ms移動(dòng)一次。滑動(dòng)窗口的格子劃分的越多,滑動(dòng)窗口的滾動(dòng)就越平滑,限流的統(tǒng)計(jì)就會(huì)越精確。

代碼如下:

/**
 * 滑窗計(jì)數(shù)器
 */
public class SliderWindowRateLimiter implements Runnable {
    //每秒允許的最大訪問數(shù)
    private final long maxVisitPerSecond;
    //將每秒時(shí)間劃分N個(gè)塊
    private final int block;
    //每個(gè)塊存儲(chǔ)的數(shù)量
    private final AtomicLong[] countPerBlock;
    //滑動(dòng)窗口劃到了哪個(gè)塊兒,可以理解為滑動(dòng)窗口的起始下標(biāo)位置
    private volatile int index;
    //目前總的數(shù)量
    private AtomicLong allCount;

    /**
     * 構(gòu)造函數(shù)
     *
     * @param block,每秒鐘劃分N個(gè)窗口
     * @param maxVisitPerSecond 每秒最大訪問數(shù)量
     */
    public SliderWindowRateLimiter(int block, long maxVisitPerSecond) {
        this.block = block;
        this.maxVisitPerSecond = maxVisitPerSecond;
        countPerBlock = new AtomicLong[block];
        for (int i = 0; i < block; i++) {
            countPerBlock[i] = new AtomicLong();
        }
        allCount = new AtomicLong(0);
    }

    /**
     * 判斷是否超過最大允許數(shù)量
     *
     * @return
     */
    public boolean isOverLimit() {
        return currentQPS() > maxVisitPerSecond;
    }

    /**
     * 獲取目前總的訪問數(shù)
     *
     * @return
     */
    public long currentQPS() {
        return allCount.get();
    }

    /**
     * 請(qǐng)求訪問進(jìn)來,判斷是否可以執(zhí)行業(yè)務(wù)邏輯
     */
    public void visit() {
        countPerBlock[index].incrementAndGet();
        allCount.incrementAndGet();

        if (isOverLimit()) {
            System.out.println(Thread.currentThread().getName() + "被限流" + ",currentQPS:" + currentQPS() + ",index:" + index);
        } else {
            System.out.println(Thread.currentThread().getName() + "執(zhí)行業(yè)務(wù)邏輯" + ",currentQPS:" + currentQPS() + ",index:" + index);
        }
    }

    /**
     * 定時(shí)執(zhí)行器,
     * 每N毫秒滑塊移動(dòng)一次,然后再設(shè)置下新滑塊的初始化數(shù)字0,然后新的請(qǐng)求會(huì)落到新的滑塊上
     * 同時(shí)總數(shù)減掉新滑塊上的數(shù)字,并且重置新的滑塊上的數(shù)量
     */
    @Override
    public void run() {
        index = (index + 1) % block;
        long val = countPerBlock[index].getAndSet(0);
        allCount.addAndGet(-val);
    }

    public static void main(String[] args) {
        SliderWindowRateLimiter sliderWindowRateLimiter = new SliderWindowRateLimiter(10, 100);

        //固定的速率移動(dòng)滑塊
        ScheduledExecutorService scheduledExecutorService = Executors.newSingleThreadScheduledExecutor();
        scheduledExecutorService.scheduleAtFixedRate(sliderWindowRateLimiter, 100, 100, TimeUnit.MILLISECONDS);

        //模擬不同速度的請(qǐng)求
        new Thread(() -> {
            while (true) {
                sliderWindowRateLimiter.visit();
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }).start();

        //模擬不同速度的請(qǐng)求
        new Thread(() -> {
            while (true) {
                sliderWindowRateLimiter.visit();
                try {
                    Thread.sleep(50);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
}

3、信號(hào)量

利用Semaphore,每隔固定速率,釋放Semaphore的資源。線程獲取到資源,則執(zhí)行業(yè)務(wù)代碼。

代碼如下:

public class SemaphoreOne {
    private static Semaphore semaphore = new Semaphore(10);

    public static void bizMethod() throws InterruptedException {
        if (!semaphore.tryAcquire()) {
            System.out.println(Thread.currentThread().getName() + "被拒絕");
            return;
        }

        System.out.println(Thread.currentThread().getName() + "執(zhí)行業(yè)務(wù)邏輯");
        Thread.sleep(500);//模擬處理業(yè)務(wù)邏輯需要1秒
        semaphore.release();
    }

    public static void main(String[] args) {

        Timer timer = new Timer();
        timer.scheduleAtFixedRate(new TimerTask() {
            @Override
            public void run() {
                semaphore.release(10);
                System.out.println("釋放所有鎖");
            }
        }, 1000, 1000);

        for (int i = 0; i < 10000; i++) {
            try {
                Thread.sleep(10);//模擬每隔10ms就有1個(gè)請(qǐng)求進(jìn)來
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            new Thread(() -> {
                try {
                    SemaphoreOne.bizMethod();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
        }
    }
}

4、令牌桶

令牌桶算法:一個(gè)存放固定容量令牌的桶,按照固定速率往桶里添加令牌,如有剩余容量則添加,沒有則放棄。如果有請(qǐng)求進(jìn)來,則需要先從桶里獲取令牌,當(dāng)桶里沒有令牌可取時(shí),則拒絕任務(wù)。

令牌桶的優(yōu)點(diǎn)是:可以改變添加令牌的速率,一旦提高速率,則可以處理突發(fā)流量。

代碼如下:

public class TokenBucket {
    /**
     * 定義的桶
     */
    public class Bucket {
        //容量
        int capacity;
        //速率,每秒放多少
        int rateCount;
        //目前token個(gè)數(shù)
        AtomicInteger curCount = new AtomicInteger(0);

        public Bucket(int capacity, int rateCount) {
            this.capacity = capacity;
            this.rateCount = rateCount;
        }

        public void put() {
            if (curCount.get() < capacity) {
                System.out.println("目前數(shù)量==" + curCount.get() + ", 我還可以繼續(xù)放");
                curCount.addAndGet(rateCount);
            }
        }

        public boolean get() {
            if (curCount.get() >= 1) {
                curCount.decrementAndGet();
                return true;
            }
            return false;
        }
    }

    @Test
    public void testTokenBucket() throws InterruptedException {

        Bucket bucket = new Bucket(5, 2);

        //固定線程,固定的速率往桶里放數(shù)據(jù),比如每秒N個(gè)
        ScheduledThreadPoolExecutor scheduledCheck = new ScheduledThreadPoolExecutor(1);
        scheduledCheck.scheduleAtFixedRate(() -> {
            bucket.put();
        }, 0, 1, TimeUnit.SECONDS);

        //先等待一會(huì)兒,讓桶里放點(diǎn)token
        Thread.sleep(6000);

        //模擬瞬間10個(gè)線程進(jìn)來拿token
        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                if (bucket.get()) {
                    System.out.println(Thread.currentThread() + "獲取到了資源");
                } else {
                    System.out.println(Thread.currentThread() + "被拒絕");
                }
            }).start();
        }

        //等待,往桶里放token
        Thread.sleep(3000);

        //繼續(xù)瞬間10個(gè)線程進(jìn)來拿token
        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                if (bucket.get()) {
                    System.out.println(Thread.currentThread() + "獲取到了資源");
                } else {
                    System.out.println(Thread.currentThread() + "被拒絕");
                }
            }).start();
        }
    }
}

5、總結(jié)

本文主要介紹了幾種限流方法:Guava RateLimiter、簡(jiǎn)單計(jì)數(shù)、滑窗計(jì)數(shù)、信號(hào)量、令牌桶,當(dāng)然,限流算法還有漏桶算法、nginx限流等等。我所寫的這些方法只是個(gè)人在實(shí)際項(xiàng)目總使用過的,或者是早年參加阿里筆試時(shí)寫過的方式。

責(zé)任編輯:趙寧寧 來源: 不焦躁的程序員
相關(guān)推薦

2021-05-14 07:45:07

Sentinel 接口限流

2024-11-05 15:02:41

2021-07-23 14:58:28

Nginx限流方案

2023-08-08 08:01:22

微服務(wù)架構(gòu)服務(wù)

2023-08-26 07:09:36

2022-10-28 18:41:53

Java服務(wù)限流

2024-11-20 15:24:49

2022-07-05 09:44:25

服務(wù)治理熔斷限流

2023-11-20 10:09:59

2021-03-30 10:46:42

SpringBoot計(jì)數(shù)器漏桶算法

2021-11-05 21:33:28

Redis數(shù)據(jù)高并發(fā)

2023-10-31 07:52:10

2022-05-19 14:14:26

go語言限流算法

2016-11-28 08:58:43

系統(tǒng)限流算法

2016-11-28 08:58:43

系統(tǒng)限流

2024-08-27 12:32:32

2024-12-25 15:44:15

2024-06-11 10:03:56

2024-08-02 19:49:41

2024-02-04 10:08:34

點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

久久一级大片| 日本动漫理论片在线观看网站| 亚洲一区二区三区四区五区午夜 | 青梅竹马是消防员在线| 日本成人中文字幕在线视频| 成人97在线观看视频| 丰满岳乱妇一区二区| 欧美电影免费观看高清完整| 成人免费在线播放视频| 国产伦精品一区二区三区免| 中文字幕免费高清在线观看| 一区在线播放| 日韩中文字幕网站| 国产白嫩美女无套久久| 亚洲午夜剧场| 日韩欧美精品网址| 屁屁影院ccyy国产第一页| 国产在线小视频| 懂色一区二区三区免费观看| 国产精品高潮呻吟久久av黑人| 免费中文字幕视频| 日韩精品欧美| 国产视频久久久久| 国产麻豆剧传媒精品国产| 国产精品videossex撒尿| 一区二区三区欧美视频| 午夜精品亚洲一区二区三区嫩草| 亚洲国产福利视频| 久久aⅴ国产欧美74aaa| 日韩av片永久免费网站| 日韩xxx高潮hd| 91精品国产91久久综合| 在线观看日韩www视频免费| 艳妇乳肉豪妇荡乳xxx| 国产一区二区三区免费观看在线 | 亚洲视频你懂的| 欧美深深色噜噜狠狠yyy| 日本高清视频在线| 国产精品66部| 91精品国产综合久久久久久丝袜| 自拍偷拍色综合| 久久精品主播| 欧洲成人在线视频| 日韩黄色a级片| 亚洲第一毛片| 国内成人精品视频| 精品少妇一二三区| 欧美黄色免费| 精品自拍视频在线观看| 永久看片925tv| 911精品美国片911久久久| 色悠悠国产精品| 欧美视频一区二区在线| 日韩av二区| 中文字幕无线精品亚洲乱码一区 | 欧美一级一区二区三区| 国产成人午夜高潮毛片| 99re在线视频上| 蜜桃视频污在线观看| 成人午夜免费av| 国产精品一区二区欧美黑人喷潮水| 99久久夜色精品国产亚洲| 国产一区二区成人久久免费影院| 92裸体在线视频网站| 国产精品无码一区二区桃花视频 | 91精品天堂| 国产成人三级在线播放| 丁香婷婷综合激情五月色| 成人高清在线观看| 亚洲av成人精品毛片| 久久亚洲精华国产精华液| 欧美国产综合视频| 成全电影播放在线观看国语| 国产精品剧情在线亚洲| www国产无套内射com| 国产美女精品写真福利视频| 欧美视频在线观看 亚洲欧| 国产精品wwwww| 久久精品资源| 日韩欧美一区二区免费| 国产精品久久久免费观看| 精品久久视频| 久久99视频免费| av资源免费观看| 青椒成人免费视频| 91九色偷拍| 日本韩国精品一区二区| 国产精品久久久一本精品 | 最新日本在线观看| 欧美三级xxx| 17c国产在线| 黄色网一区二区| 中文国产成人精品| 免费在线视频一区二区| 日日摸夜夜添夜夜添精品视频| 成人久久一区二区| 色一情一乱一区二区三区| 亚洲国产精品激情在线观看| 久久久久久久久网| 精品国产免费人成网站| 欧美一区二区三级| 国产呦小j女精品视频| 91精品观看| 91高潮精品免费porn| 国产精品久久久久久久久久久久久久久久久久 | 久久久久久欧美精品色一二三四| 亚洲免费视频一区二区三区| 亚洲动漫第一页| 爱爱爱爱免费视频| 欧美大奶一区二区| 欧美成人免费在线观看| 中文字幕在线天堂| 成人午夜电影网站| 性欧美18一19内谢| 日韩pacopacomama| 亚洲高清久久网| 紧身裙女教师波多野结衣| 久久午夜精品| 国产精品国模大尺度私拍| av播放在线观看| 色噜噜狠狠一区二区三区果冻| 91福利视频免费观看| 久久国产影院| 国产精品第一页在线| 日本成人一区| 亚洲国产乱码最新视频 | 91丨九色丨国产丨porny| 欧美 国产 精品| 六九午夜精品视频| 国产亚洲xxx| 你懂的国产在线| 不卡视频在线看| 精品久久久无码人妻字幂| 懂色aⅴ精品一区二区三区| 亚洲欧美www| 羞羞影院体验区| 99久久亚洲一区二区三区青草 | 最新日韩免费视频| 日本不卡免费在线视频| 日韩高清国产精品| 激情开心成人网| 亚洲美女www午夜| 男女啊啊啊视频| 91一区二区在线| www.99热这里只有精品| 岛国成人av| 91精品国产91久久久久| 色婷婷激情五月| 欧美日韩在线看| 亚洲中文字幕一区| 国产欧美一区二区三区国产幕精品| 国产成人亚洲欧美| 国产精品25p| 日韩高清av一区二区三区| 视频一区二区三区四区五区| 91美女片黄在线| 成人羞羞国产免费网站| 欧美日韩123| 国产精品吴梦梦| 久久精品视频免费看| 91精品国产入口在线| 青青草原国产视频| 成人手机电影网| 大陆极品少妇内射aaaaa| 先锋影音国产精品| 国产精品成人品| 欧美成人hd| 日韩精品一区二区三区在线 | 国产一区二区中文字幕免费看| 僵尸再翻生在线观看| 亚洲免费电影在线观看| 中文字幕永久在线视频| 亚洲蜜桃精久久久久久久| 91精品又粗又猛又爽| 国产模特精品视频久久久久| 日本电影一区二区三区| www.欧美视频| 97色在线播放视频| av网在线观看| 日韩免费看网站| 欧产日产国产69| 中文字幕一区二区三区精华液 | 国产亲近乱来精品视频 | 91女人视频在线观看| 亚洲成人av免费看| 国产精品草草| 日本欧洲国产一区二区| 日本一区二区乱| 欧美重口另类videos人妖| 日本在线免费网| 亚洲精品电影网站| 一区二区三区精彩视频| 午夜激情久久久| 毛片视频免费播放| 丁香另类激情小说| 亚洲福利精品视频| 黄色日韩在线| 亚洲精品视频一二三| www.亚洲一二| 国产色视频一区| 高清在线视频不卡| 久久视频免费在线播放| 深夜福利在线看| 日韩视频在线永久播放| 国产在线一级片| 亚洲成av人片在线观看无码| 黄色一级片一级片| 2023国产精品| 性猛交╳xxx乱大交| 精品中文字幕一区二区 | 国产一区二区三区四区五区入口| 日日橹狠狠爱欧美超碰| 中文字幕人成人乱码| 日本一区二区三区免费看| 盗摄系列偷拍视频精品tp| 成人免费午夜电影| 日韩欧美一区二区三区免费观看 | 97久久精品人人爽人人爽蜜臀 | 韩国精品福利一区二区三区| 国产精品亚洲片夜色在线| wwwww亚洲| 九九九久久国产免费| 午夜小视频在线| 国产香蕉97碰碰久久人人| 视频三区在线观看| 亚洲国语精品自产拍在线观看| 99在线精品视频免费观看20| 欧美日韩免费高清一区色橹橹| 91精品国产高清一区二区三密臀| 亚洲一二三四在线观看| 欧美卡一卡二卡三| 国产精品乱码人人做人人爱| 亚洲av毛片基地| 久久精品在这里| 免费黄色在线视频| 久久人人爽人人爽| 少妇真人直播免费视频| 91丨九色丨国产丨porny| 亚洲精品中文字幕在线播放| 国产999精品久久久久久| 人妻巨大乳一二三区| 国产精品综合视频| 亚洲精品中文字幕乱码无线| 极品少妇xxxx精品少妇偷拍| 九色porny自拍| 精品综合久久久久久8888| 男人的天堂最新网址| 蜜臀av国产精品久久久久 | 欧美国产视频| 国产日韩欧美视频在线| 深夜福利亚洲| 91老司机精品视频| 欧美第一在线视频| av电影成人| 久久av国产紧身裤| 欧美一区国产一区| 欧州一区二区| 香蕉视频在线网址| 国产专区一区| 国产主播在线看| 蜜臀va亚洲va欧美va天堂 | 男人插女人视频在线观看| 激情久久综合| 国产免费一区二区三区视频| 日本在线不卡视频一二三区| 在线黄色免费看| 国产成人综合在线播放| 久久久久久婷婷| 久久久91精品国产一区二区精品| 亚洲一区 欧美| 亚洲欧美视频一区| 国产午夜久久久| 色综合久久久久久久久| 中文字幕一二区| 日韩一区二区麻豆国产| 色综合久久网女同蕾丝边| 在线视频欧美日韩精品| 亚洲丝袜精品| 欧美在线视频一区| 成人国产网站| 国产不卡一区二区三区在线观看| 性人久久久久| 国产精品无码乱伦| 亚洲一区中文| 在线播放av中文字幕| www..com久久爱| 成人免费视频入口| 亚洲国产另类av| 中文字幕精品在线观看| 精品国产髙清在线看国产毛片| 欧美成人片在线| 美女啪啪无遮挡免费久久网站| 九色porny自拍视频在线观看| 国产精品免费网站| 国产一区二区三区亚洲| 一区二区三区四区| 国产欧美一区二区色老头 | 中国女人真人一级毛片| 精品久久五月天| 在线免费看黄网站| 97人人做人人爱| 欧美一区一区| 亚州欧美一区三区三区在线| 亚洲国产免费| 初高中福利视频网站| 国产欧美日韩久久| 五月天婷婷综合网| 欧美一级高清片在线观看| 日韩三级电影网| 久久久久久香蕉网| 激情视频亚洲| 亚洲国产一区二区三区在线| 国产欧美日韩亚洲一区二区三区| 永久av免费在线观看| 国产三区在线成人av| 日韩av片在线播放| 欧美不卡一区二区三区| 麻豆视频在线| 日韩免费在线播放| 欧美一性一交| 日本阿v视频在线观看| 国产麻豆精品theporn| 超碰人人人人人人人| 色网综合在线观看| 亚洲av激情无码专区在线播放| 精品视频9999| 日韩视频一二区| 一本二本三本亚洲码 | 日韩精品国产一区| 亚洲人123区| 国产精品熟女久久久久久| 一区二区三区 在线观看视| 亚洲精品mv| 久久精品99久久| 国产一区二区三区久久久久久久久 | 亚洲一区三区| 麻豆久久久久久久| 女人黄色一级片| 欧美四级电影在线观看| 成人在线观看一区| 国产精品99久久久久久久久 | 免费看欧美一级片| 国产91精品入口| 精品在线视频观看| 亚洲аv电影天堂网| 岛国毛片av在线| 国产另类自拍| 国产精品毛片在线| 国产福利短视频| 一本一道综合狠狠老| 麻豆app在线观看| 国产精品久久久久久av福利| 欧美亚洲国产精品久久| 91女神在线观看| 18成人在线观看| 国产成人久久精品77777综合 | 亚洲精品视频在线观看免费视频| 精品捆绑美女sm三区| ririsao久久精品一区| 精品国产免费久久久久久尖叫| 亚洲神马久久| 在线免费观看视频| 在线不卡a资源高清| 五月花成人网| 久久国产一区二区| 欧美aaaaa成人免费观看视频| 午夜黄色福利视频| 日韩精品一区二区三区中文不卡| 成人在线免费观看黄色| 久久手机视频| 久久国产精品无码网站| 精品欧美一区二区久久久久| 日韩av在线免费播放| 久久天堂av| 水蜜桃在线免费观看| av福利精品导航| japanese国产在线观看| 久久偷看各类女兵18女厕嘘嘘| 粉嫩久久久久久久极品| 中文字幕乱码人妻综合二区三区 | 国产91精品精华液一区二区三区| 日本午夜小视频| 中文字幕欧美国内| 亚洲码欧美码一区二区三区| 黄www在线观看| 一区二区中文视频| 亚洲 国产 欧美 日韩| 国产女同一区二区| 亚洲欧洲综合| 亚洲精品国产精品乱码在线观看| 精品日韩一区二区三区免费视频| a日韩av网址| 永久免费看av| 中文字幕va一区二区三区| 秋霞av鲁丝片一区二区| 国产精品揄拍一区二区| 国产日韩1区| 免费在线观看一级片| 亚洲天堂免费观看| 超碰在线亚洲|