精品欧美一区二区三区在线观看 _久久久久国色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)

日韩在线观看电影完整版高清免费悬疑悬疑| 久久免费电影| 麻豆视频观看网址久久| 久久精品久久久久| 免费看黄色片的网站| 国产污视频在线播放| 国产女同性恋一区二区| 91精品天堂| 黄色网址中文字幕| 国产一区日韩欧美| 一区二区三区www| 日本一级大毛片a一| 欧美日韩免费观看视频| 一区二区三区四区精品在线视频| 欧美日韩高清免费| 精品国产99久久久久久宅男i| 国产亚洲一级| 欧美成年人视频| 亚洲久久久久久久| www.成人网| 欧美日韩极品在线观看一区| 欧美激情视频免费看| 尤物网在线观看| 99视频有精品| 99久久久久国产精品免费| 日批视频免费在线观看| 亚洲午夜91| 久久视频在线直播| 丁香激情五月少妇| 精品一区在线| 日韩不卡在线观看| 中文字幕乱妇无码av在线| 高清av一区二区三区| 狠狠色狠狠色综合日日小说| 日韩欧美视频免费在线观看| 色综合久久久久综合一本到桃花网| av不卡在线播放| 成人3d动漫一区二区三区91| 一本色道久久综合熟妇| 在线亚洲一区| 国产+成+人+亚洲欧洲| 日本高清不卡免费| 日韩在线高清| 最新69国产成人精品视频免费| 韩国无码一区二区三区精品| 成人免费在线电影网| 日韩精品在线一区二区| 黄色一级片免费播放| 日本一区二区中文字幕| 欧美乱妇20p| 亚洲一区二区在线视频观看| 六九午夜精品视频| 欧美日韩激情一区二区三区| 爱情岛论坛亚洲首页入口章节| 欧美成a人片在线观看久| 黑人巨大精品欧美一区二区免费 | 国产永久免费网站| 亚洲精品**中文毛片| 五月综合激情网| 5月婷婷6月丁香| 精精国产xxx在线视频app| 精品久久久久久亚洲国产300| 少妇高潮毛片色欲ava片| free性m.freesex欧美| 狠狠色狠狠色综合日日五| 欧洲av无码放荡人妇网站| 欧美www.| 91精品国产麻豆国产自产在线| 久久精品一二三四| 国产精品sss在线观看av| 亚洲成人三级在线| wwwwxxxx国产| 色135综合网| 久久亚洲国产精品| 久久精品久久精品久久| 亚欧成人精品| 国产精品夜色7777狼人| 国产成人精品无码高潮| 99久免费精品视频在线观看| 日本不卡一二三区| lutube成人福利在线观看| 亚洲欧美电影一区二区| www..com日韩| 影音成人av| 日韩欧美一区二区在线视频| 免费的av网站| 日韩在线欧美| 91wwwcom在线观看| 亚洲一级视频在线观看| 成人手机在线视频| 三区精品视频观看| 色呦呦在线资源| 色综合久久久网| 午夜福利123| 日韩在线影视| 中文字幕不卡av| 一级aaa毛片| 欧美aⅴ一区二区三区视频| 91亚洲永久免费精品| 日本韩国精品一区二区| 亚洲天堂2016| 91视频最新入口| 精品午夜视频| 中文字幕v亚洲ⅴv天堂| 91精品国产乱码久久久张津瑜| 美腿丝袜一区二区三区| 国产一级特黄a大片99| 欧美三级黄网| 色综合天天在线| av在线天堂网| 久久影院一区| 欧美亚洲国产日韩2020| 国产黄频在线观看| 中文字幕不卡在线观看| 男人日女人视频网站| 日韩一区二区三区高清在线观看| 亚洲欧美精品在线| 欧美交换国产一区内射| 久久国产人妖系列| 日本不卡一区二区三区在线观看 | 国产狼人综合免费视频| 天堂中文在线资源| 一区二区三区四区高清精品免费观看| 亚洲色图久久久| 老司机成人在线| 欧美日韩国产成人在线观看| 亚洲天堂男人网| 久久久亚洲精品石原莉奈 | 性国裸体高清亚洲| 精品成人在线观看| 麻豆成人在线视频| 国产在线视频一区二区三区| 手机看片福利永久国产日韩| 欧美电影免费观看| 日韩精品在线观看视频| 日本午夜精品理论片a级app发布| 国产又黄又大久久| 中日韩在线视频| 日韩一区中文| 日韩在线激情视频| 在线观看毛片网站| 国产精品丝袜久久久久久app| 欧美污视频网站| 亚洲自拍都市欧美小说| 欧美亚洲成人网| 五月天婷婷视频| 欧美日韩国产综合视频在线观看中文| 性欧美18—19sex性高清| 伊人狠狠色j香婷婷综合| 都市激情久久久久久久久久久| 性欧美ⅴideo另类hd| 日韩欧美激情一区| 久久久精品99| 成人精品视频.| 国产精品自拍片| 中文字幕av一区二区三区人| 日本一欧美一欧美一亚洲视频| 国产在线视频福利| 欧美视频在线一区二区三区 | 性高潮久久久久久久| 久久亚洲国产精品一区二区| 日韩精品在在线一区二区中文| 成人免费网站www网站高清| 中文字幕成人在线| 国产农村妇女毛片精品久久| 亚洲一区二区三区视频在线播放| 国产精品久久无码| 奇米色一区二区| 午夜啪啪福利视频| av一级亚洲| 日韩免费av片在线观看| 黄色在线视频网站| 精品国产网站在线观看| 中文字幕免费在线观看视频| 亚洲国产岛国毛片在线| www.久久com| 一区二区动漫| 五月天亚洲综合小说网| 精品国产鲁一鲁****| 777午夜精品福利在线观看| shkd中文字幕久久在线观看| 日韩一区二区三区视频在线观看| 西西44rtwww国产精品| 国产女主播视频一区二区| 久草福利在线观看| 久久婷婷亚洲| 天堂а√在线中文在线| 一区二区小说| 亚洲www视频| 亚洲性受xxx喷奶水| 久久精品国产99国产精品澳门| 天天操天天舔天天干| 欧美日韩午夜在线视频| 国产精品6666| 亚洲欧洲综合另类在线 | 中文字幕一区二区视频| 日本国产在线视频| 久久国产成人午夜av影院| 999在线观看视频| 婷婷综合社区| 欧美日本韩国在线| 91国内精品| 成人免费午夜电影| 亚洲www免费| 久久人人爽人人| 国产原创视频在线观看| 亚洲欧美资源在线| 免费看av毛片| 5566中文字幕一区二区电影| 无码免费一区二区三区| 亚洲制服丝袜在线| 三级黄色录像视频| 久久九九久久九九| 免费a v网站| 国产不卡视频一区| 在线不卡一区二区三区| 久久久久久穴| 欧美s码亚洲码精品m码| 欧美日韩亚洲国产精品| 综合视频在线观看| 欧美日韩亚洲在线观看| 欧美日韩精品免费看| 国内露脸中年夫妇交换精品| 亚洲一区二区三区xxx视频| 国产精品麻豆成人av电影艾秋| 欧美与黑人午夜性猛交久久久| h片在线观看视频免费免费| 久99久在线视频| 成人短视频在线| xxxxxxxxx欧美| 日韩精品成人av| 色一区av在线| 中文日本在线观看| 中国人与牲禽动交精品| 国产在线一二三区| 一区二区三区高清国产| av在线二区| 中文字幕亚洲二区| 91社区在线高清| 中文字幕av一区二区| 亚洲精品承认| 久久久成人av| gogo在线高清视频| 九九热视频这里只有精品| 亚洲小说区图片区都市| 欧美国产亚洲视频| 欧美14一18处毛片| 高清一区二区三区四区五区| 蜜桃麻豆av在线| 日av在线播放中文不卡| 欧美日韩精品免费观看视完整| 国产精品国语对白| 四虎精品一区二区免费| 91在线看www| 国产欧美三级电影| 久久久久久艹| 国产探花一区| 亚洲欧美一二三| 欧美日韩p片| 久久网站免费视频| 奇米精品一区二区三区四区| 午夜一级免费视频| 国产成人免费视频网站| 中文字幕在线国产| 91麻豆精品一区二区三区| 一级片视频免费看| 国产精品成人午夜| 国产一级做a爰片在线看免费| 精品久久香蕉国产线看观看亚洲| 久久久久久在线观看| 欧美美女一区二区三区| www.桃色av嫩草.com| 日韩电影第一页| 一区二区三区视频在线观看视频| 欧美成人午夜激情| 三级成人黄色影院| 91在线中文字幕| 综合综合综合综合综合网| 一区二区在线观看网站| 亚洲第一精品影视| 日本久久久久久久久久久久| 国产麻豆精品theporn| 久久久久麻豆v国产精华液好用吗| 欧美激情一区在线观看| 久久久久免费看| 在线观看www91| 亚洲第一页视频| 亚洲视频在线观看| 青草视频在线免费直播| 国产精品99久久久久久白浆小说| 韩国三级大全久久网站| 欧美日韩国产高清视频| 欧美日本一区| 性欧美videossex精品| 不卡一区在线观看| 91麻豆精品久久毛片一级| 午夜日韩在线电影| 97超碰人人模人人人爽人人爱| 亚洲精品乱码久久久久久按摩观| 国产黄大片在线观看画质优化| 欧美一区二区三区精品电影| 久久精品九色| 性欧美.com| 亚洲一区亚洲| 欧美极品jizzhd欧美仙踪林| 国产精品毛片大码女人| 亚洲精品男人天堂| 欧美v亚洲v综合ⅴ国产v| av网站无病毒在线| 欧美一区二区大胆人体摄影专业网站| 日本亚州欧洲精品不卡| 亚洲一区精彩视频| 日日摸夜夜添夜夜添亚洲女人| 国产精品成人无码专区| 亚洲黄色av一区| 一区二区三区免费在线视频| 亚洲日韩欧美视频一区| 精品捆绑调教一区二区三区| 97在线中文字幕| 91tv官网精品成人亚洲| 91极品尤物在线播放国产| 久久精品夜色噜噜亚洲a∨| 日韩久久久久久久久| 日韩美女一区二区三区四区| 美女免费久久| 国产在线拍偷自揄拍精品| 少妇精品久久久一区二区三区| 久久久亚洲国产精品| 国产成人8x视频一区二区| 少妇人妻丰满做爰xxx| 欧美日韩成人在线一区| av资源网站在线观看| 国产精品视频999| 日本黄色精品| 在线观看免费视频高清游戏推荐| 国产日产亚洲精品系列| 亚洲黄网在线观看| 国产一区二区三区在线看| 写真福利精品福利在线观看| 日韩精品一区二区三区丰满| 青草av.久久免费一区| 亚洲精品91在线| 精品视频在线免费观看| 婷婷成人激情| 亚洲自拍偷拍福利| 国产精品成人一区二区网站软件| 在线播放国产视频| 亚洲成人免费影院| 天堂a中文在线| 国产精品成人一区二区三区吃奶| 国产精品美女久久久久久不卡| 黄色高清无遮挡| 欧美国产精品一区| 97人妻精品一区二区三区软件| 久久伊人精品一区二区三区| 日韩中文字幕| 国产二级片在线观看| 久久精品视频在线看| 中文在线a天堂| 欧美www在线| 欧美激情影院| 国产精品天天av精麻传媒| 国产精品美女久久久久久久久 | 日韩成人小视频| 欧美羞羞视频| 亚洲在线色站| 国产91精品露脸国语对白| www.av麻豆| 在线观看欧美成人| 久久的色偷偷| www国产黄色| 国产精品九色蝌蚪自拍| www.天堂在线| 日本成人激情视频| 久久在线视频| 污污免费在线观看| 欧美视频一区二| 三级资源在线| 日本视频一区在线观看| 精品影视av免费| 福利一区二区三区四区| 有码中文亚洲精品| 亚洲日本va中文字幕| 精品久久久久av| 亚洲男人的天堂在线aⅴ视频| 亚洲 欧美 激情 小说 另类| 国产欧美精品va在线观看| 伊人久久久大香线蕉综合直播 | 国产欧美最新羞羞视频在线观看| 欧美激情偷拍| 香蕉视频久久久| 欧美成人女星排名| 精品成人av| 国产69精品久久久久久久| 中文欧美字幕免费| 日韩一区二区三区在线观看视频| 国产精品自拍偷拍| 国产日韩视频| 欧美黄色免费观看| 一区二区欧美日韩视频|