談?wù)勏蘖魉惴ǖ膸追N實現(xiàn)

占小狼,轉(zhuǎn)載請注明原創(chuàng)出處泪蔫,謝謝箭券!

保障服務(wù)穩(wěn)定的三大利器:熔斷降級睡榆、服務(wù)限流和故障模擬瓮栗。今天和大家談?wù)勏蘖魉惴ǖ膸追N實現(xiàn)方式,本文所說的限流并非是Nginx層面的限流沦泌,而是業(yè)務(wù)代碼中的邏輯限流辽慕。

為什么需要限流

按照服務(wù)的調(diào)用方,可以分為以下幾種類型服務(wù)

1赦肃、與用戶打交道的服務(wù)

比如web服務(wù)溅蛉、對外API,這種類型的服務(wù)有以下幾種可能導致機器被拖垮:

  • 用戶增長過快(這是好事)
  • 因為某個熱點事件(微博熱搜)
  • 競爭對象爬蟲
  • 惡意的刷單

這些情況都是無法預知的他宛,不知道什么時候會有10倍甚至20倍的流量打進來船侧,如果真碰上這種情況,擴容是根本來不及的(彈性擴容都是虛談厅各,一秒鐘你給我擴一下試試)

2镜撩、對內(nèi)的RPC服務(wù)

一個服務(wù)A的接口可能被BCDE多個服務(wù)進行調(diào)用,在B服務(wù)發(fā)生突發(fā)流量時队塘,直接把A服務(wù)給調(diào)用掛了袁梗,導致A服務(wù)對CDE也無法提供服務(wù)。
這種情況時有發(fā)生憔古,解決方案有兩種:
1遮怜、每個調(diào)用方采用線程池進行資源隔離
2、使用限流手段對每個調(diào)用方進行限流

限流算法實現(xiàn)

常見的限流算法有:計數(shù)器鸿市、令牌桶锯梁、漏桶。

1焰情、計數(shù)器算法

采用計數(shù)器實現(xiàn)限流有點簡單粗暴陌凳,一般我們會限制一秒鐘的能夠通過的請求數(shù),比如限流qps為100内舟,算法的實現(xiàn)思路就是從第一個請求進來開始計時合敦,在接下去的1s內(nèi),每來一個請求验游,就把計數(shù)加1充岛,如果累加的數(shù)字達到了100,那么后續(xù)的請求就會被全部拒絕批狱。等到1s結(jié)束后裸准,把計數(shù)恢復成0,重新開始計數(shù)赔硫。

具體的實現(xiàn)可以是這樣的:對于每次服務(wù)調(diào)用炒俱,可以通過AtomicLong#incrementAndGet()方法來給計數(shù)器加1并返回最新值,通過這個最新值和閾值進行比較。

這種實現(xiàn)方式权悟,相信大家都知道有一個弊端:如果我在單位時間1s內(nèi)的前10ms砸王,已經(jīng)通過了100個請求,那后面的990ms峦阁,只能眼巴巴的把請求拒絕谦铃,我們把這種現(xiàn)象稱為“突刺現(xiàn)象”

2、漏桶算法

為了消除"突刺現(xiàn)象"榔昔,可以采用漏桶算法實現(xiàn)限流驹闰,漏桶算法這個名字就很形象,算法內(nèi)部有一個容器撒会,類似生活用到的漏斗嘹朗,當請求進來時,相當于水倒入漏斗诵肛,然后從下端小口慢慢勻速的流出屹培。不管上面流量多大,下面流出的速度始終保持不變怔檩。

不管服務(wù)調(diào)用方多么不穩(wěn)定褪秀,通過漏桶算法進行限流,每10毫秒處理一次請求薛训。因為處理的速度是固定的媒吗,請求進來的速度是未知的,可能突然進來很多請求许蓖,沒來得及處理的請求就先放在桶里蝴猪,既然是個桶,肯定是有容量上限膊爪,如果桶滿了,那么新進來的請求就丟棄嚎莉。

圖來自網(wǎng)上

在算法實現(xiàn)方面米酬,可以準備一個隊列,用來保存請求趋箩,另外通過一個線程池(ScheduledExecutorService)來定期從隊列中獲取請求并執(zhí)行赃额,可以一次性獲取多個并發(fā)執(zhí)行。

這種算法叫确,在使用過后也存在弊端:無法應(yīng)對短時間的突發(fā)流量跳芳。

3、令牌桶算法

從某種意義上講竹勉,令牌桶算法是對漏桶算法的一種改進飞盆,桶算法能夠限制請求調(diào)用的速率,而令牌桶算法能夠在限制調(diào)用的平均速率的同時還允許一定程度的突發(fā)調(diào)用。

在令牌桶算法中吓歇,存在一個桶孽水,用來存放固定數(shù)量的令牌。算法中存在一種機制城看,以一定的速率往桶中放令牌女气。每次請求調(diào)用需要先獲取令牌,只有拿到令牌测柠,才有機會繼續(xù)執(zhí)行炼鞠,否則選擇選擇等待可用的令牌、或者直接拒絕轰胁。

放令牌這個動作是持續(xù)不斷的進行谒主,如果桶中令牌數(shù)達到上限,就丟棄令牌软吐,所以就存在這種情況瘩将,桶中一直有大量的可用令牌,這時進來的請求就可以直接拿到令牌執(zhí)行凹耙,比如設(shè)置qps為100姿现,那么限流器初始化完成一秒后,桶中就已經(jīng)有100個令牌了肖抱,這時服務(wù)還沒完全啟動好备典,等啟動完成對外提供服務(wù)時,該限流器可以抵擋瞬時的100個請求意述。所以提佣,只有桶中沒有令牌時,請求才會進行等待荤崇,最后相當于以一定的速率執(zhí)行拌屏。

實現(xiàn)思路:可以準備一個隊列,用來保存令牌术荤,另外通過一個線程池定期生成令牌放到隊列中倚喂,每來一個請求,就從隊列中獲取一個令牌瓣戚,并繼續(xù)執(zhí)行端圈。

通過Google開源的guava包,我們可以很輕松的創(chuàng)建一個令牌桶算法的限流器子库。

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

通過RateLimiter類的create方法舱权,創(chuàng)建限流器。

public class RateLimiterMain {
    public static void main(String[] args) {
        RateLimiter rateLimiter = RateLimiter.create(10);
        for (int i = 0; i < 10; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    rateLimiter.acquire()
                    System.out.println("pass");
                }
            }).start();
        }
    }
}

其實Guava提供了多種create方法仑嗅,方便創(chuàng)建適合各種需求的限流器宴倍。在上述例子中张症,創(chuàng)建了一個每秒生成10個令牌的限流器,即100ms生成一個啊楚,并最多保存10個令牌吠冤,多余的會被丟棄。

rateLimiter提供了acquire()和tryAcquire()接口
1恭理、使用acquire()方法拯辙,如果沒有可用令牌,會一直阻塞直到有足夠的令牌颜价。
2涯保、使用tryAcquire()方法,如果沒有可用令牌周伦,就直接返回false夕春。
3、使用tryAcquire()帶超時時間的方法专挪,如果沒有可用令牌及志,就會判斷在超時時間內(nèi)是否可以等到令牌,如果不能寨腔,就返回false速侈,如果可以,就阻塞等待迫卢。

集群限流

前面討論的幾種算法都屬于單機限流的范疇倚搬,但是業(yè)務(wù)需求五花八門,簡單的單機限流乾蛤,根本無法滿足他們每界。

比如為了限制某個資源被每個用戶或者商戶的訪問次數(shù),5s只能訪問2次家卖,或者一天只能調(diào)用1000次眨层,這種需求,單機限流是無法實現(xiàn)的上荡,這時就需要通過集群限流進行實現(xiàn)谐岁。

如何實現(xiàn)?
為了控制訪問次數(shù)艇潭,肯定需要一個計數(shù)器挤牛,而且這個計數(shù)器只能保存在第三方服務(wù),比如redis。

大概思路:每次有相關(guān)操作的時候看疗,就向redis服務(wù)器發(fā)送一個incr命令,比如需要限制某個用戶訪問/index接口的次數(shù)辖试,只需要拼接用戶id和接口名生成redis的key循签,每次該用戶訪問此接口時,只需要對這個key執(zhí)行incr命令,在這個key帶上過期時間尤蛮,就可以實現(xiàn)指定時間的訪問頻率媳友。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市产捞,隨后出現(xiàn)的幾起案子醇锚,更是在濱河造成了極大的恐慌,老刑警劉巖坯临,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件焊唬,死亡現(xiàn)場離奇詭異,居然都是意外死亡看靠,警方通過查閱死者的電腦和手機赶促,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來挟炬,“玉大人鸥滨,你說我怎么就攤上這事“妫” “怎么了婿滓?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長泊脐。 經(jīng)常有香客問我空幻,道長,這世上最難降的妖魔是什么容客? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任秕铛,我火速辦了婚禮,結(jié)果婚禮上缩挑,老公的妹妹穿的比我還像新娘但两。我一直安慰自己,他們只是感情好供置,可當我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布谨湘。 她就那樣靜靜地躺著,像睡著了一般芥丧。 火紅的嫁衣襯著肌膚如雪紧阔。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天续担,我揣著相機與錄音擅耽,去河邊找鬼。 笑死物遇,一個胖子當著我的面吹牛乖仇,可吹牛的內(nèi)容都是我干的憾儒。 我是一名探鬼主播,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼乃沙,長吁一口氣:“原來是場噩夢啊……” “哼起趾!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起警儒,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤训裆,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后冷蚂,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體缭保,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年蝙茶,在試婚紗的時候發(fā)現(xiàn)自己被綠了艺骂。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡隆夯,死狀恐怖钳恕,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情蹄衷,我是刑警寧澤忧额,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站愧口,受9級特大地震影響睦番,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜耍属,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一托嚣、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧厚骗,春花似錦示启、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至冲秽,卻和暖如春舍咖,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背锉桑。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工谎仲, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人刨仑。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓郑诺,卻偏偏與公主長得像,于是被迫代替她去往敵國和親杉武。 傳聞我的和親對象是個殘疾皇子辙诞,可洞房花燭夜當晚...
    茶點故事閱讀 42,792評論 2 345

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