摘要:令牌桶算法對于很多應(yīng)用場景來說,除了要求能夠限制數(shù)據(jù)的平均傳輸速率外,還要求允許某種程度的突發(fā)傳輸。使用以及源碼解析開源工具包提供了限流工具類,該類基于令牌桶算法實(shí)現(xiàn)流量限制,使用十分方便,而且十分高效。
前言
在開發(fā)高并發(fā)系統(tǒng)時(shí)有三把利器用來保護(hù)系統(tǒng):緩存、降級和限流
緩存 緩存的目的是提升系統(tǒng)訪問速度和增大系統(tǒng)處理容量
降級 降級是當(dāng)服務(wù)出現(xiàn)問題或者影響到核心流程時(shí),需要暫時(shí)屏蔽掉,待高峰或者問題解決后再打開
限流 限流的目的是通過對并發(fā)訪問/請求進(jìn)行限速,或者對一個(gè)時(shí)間窗口內(nèi)的請求進(jìn)行限速來保護(hù)系統(tǒng),一旦達(dá)到限制速率則可以拒絕服務(wù)、排隊(duì)或等待、降級等處理
常用的限流算法 漏桶算法漏桶算法思路很簡單,水(請求)先進(jìn)入到漏桶里,漏桶以一定的速度出水,當(dāng)水流入速度過大會直接溢出,可以看出漏桶算法能強(qiáng)行限制數(shù)據(jù)的傳輸速率。令牌桶算法
對于很多應(yīng)用場景來說,除了要求能夠限制數(shù)據(jù)的平均傳輸速率外,還要求允許某種程度的突發(fā)傳輸。這時(shí)候漏桶算法可能就不合適了,令牌桶算法更為適合。如圖所示,令牌桶算法的原理是系統(tǒng)會以一個(gè)恒定的速度往桶里放入令牌,而如果請求需要被處理,則需要先從桶里獲取一個(gè)令牌,當(dāng)桶里沒有令牌可取時(shí),則拒絕服務(wù)。RateLimiter使用以及源碼解析
Google開源工具包Guava提供了限流工具類RateLimiter,該類基于令牌桶算法實(shí)現(xiàn)流量限制,使用十分方便,而且十分高效。RateLimiter使用
首先簡單介紹下RateLimiter的使用,
public void testAcquire() { RateLimiter limiter = RateLimiter.create(1); for(int i = 1; i < 10; i = i + 2 ) { double waitTime = limiter.acquire(i); System.out.println("cutTime=" + System.currentTimeMillis() + " acq:" + i + " waitTime:" + waitTime); } }
輸出結(jié)果:
cutTime=1535439657427 acq:1 waitTime:0.0 cutTime=1535439658431 acq:3 waitTime:0.997045 cutTime=1535439661429 acq:5 waitTime:2.993028 cutTime=1535439666426 acq:7 waitTime:4.995625 cutTime=1535439673426 acq:9 waitTime:6.999223
首先通過RateLimiter.create(1);創(chuàng)建一個(gè)限流器,參數(shù)代表每秒生成的令牌數(shù),通過limiter.acquire(i);來以阻塞的方式獲取令牌,當(dāng)然也可以通過tryAcquire(int permits, long timeout, TimeUnit unit)來設(shè)置等待超時(shí)時(shí)間的方式獲取令牌,如果超timeout為0,則代表非阻塞,獲取不到立即返回。
從輸出來看,RateLimiter支持預(yù)消費(fèi),比如在acquire(5)時(shí),等待時(shí)間是3秒,是上一個(gè)獲取令牌時(shí)預(yù)消費(fèi)了3個(gè)兩排,固需要等待3*1秒,然后又預(yù)消費(fèi)了5個(gè)令牌,以此類推
RateLimiter通過限制后面請求的等待時(shí)間,來支持一定程度的突發(fā)請求(預(yù)消費(fèi)),在使用過程中需要注意這一點(diǎn),具體實(shí)現(xiàn)原理后面再分析。
RateLimiter實(shí)現(xiàn)原理Guava有兩種限流模式,一種為穩(wěn)定模式(SmoothBursty:令牌生成速度恒定),一種為漸進(jìn)模式(SmoothWarmingUp:令牌生成速度緩慢提升直到維持在一個(gè)穩(wěn)定值) 兩種模式實(shí)現(xiàn)思路類似,主要區(qū)別在等待時(shí)間的計(jì)算上,本篇重點(diǎn)介紹SmoothBursty
通過調(diào)用RateLimiter的create接口來創(chuàng)建實(shí)例,實(shí)際是調(diào)用的SmoothBuisty穩(wěn)定模式創(chuàng)建的實(shí)例。
public static RateLimiter create(double permitsPerSecond) { return create(permitsPerSecond, SleepingStopwatch.createFromSystemTimer()); } static RateLimiter create(double permitsPerSecond, SleepingStopwatch stopwatch) { RateLimiter rateLimiter = new SmoothBursty(stopwatch, 1.0 /* maxBurstSeconds */); rateLimiter.setRate(permitsPerSecond); return rateLimiter; }
SmoothBursty中的兩個(gè)構(gòu)造參數(shù)含義:
SleepingStopwatch:guava中的一個(gè)時(shí)鐘類實(shí)例,會通過這個(gè)來計(jì)算時(shí)間及令牌
maxBurstSeconds:官方解釋,在ReteLimiter未使用時(shí),最多保存幾秒的令牌,默認(rèn)是1
在解析SmoothBursty原理前,重點(diǎn)解釋下SmoothBursty中幾個(gè)屬性的含義
/** * The work (permits) of how many seconds can be saved up if this RateLimiter is unused? * 在RateLimiter未使用時(shí),最多存儲幾秒的令牌 * */ final double maxBurstSeconds; /** * The currently stored permits. * 當(dāng)前存儲令牌數(shù) */ double storedPermits; /** * The maximum number of stored permits. * 最大存儲令牌數(shù) = maxBurstSeconds * stableIntervalMicros(見下文) */ double maxPermits; /** * The interval between two unit requests, at our stable rate. E.g., a stable rate of 5 permits * per second has a stable interval of 200ms. * 添加令牌時(shí)間間隔 = SECONDS.toMicros(1L) / permitsPerSecond;(1秒/每秒的令牌數(shù)) */ double stableIntervalMicros; /** * The time when the next request (no matter its size) will be granted. After granting a request, * this is pushed further in the future. Large requests push this further than small requests. * 下一次請求可以獲取令牌的起始時(shí)間 * 由于RateLimiter允許預(yù)消費(fèi),上次請求預(yù)消費(fèi)令牌后 * 下次請求需要等待相應(yīng)的時(shí)間到nextFreeTicketMicros時(shí)刻才可以獲取令牌 */ private long nextFreeTicketMicros = 0L; // could be either in the past or future
接下來介紹幾個(gè)關(guān)鍵函數(shù)
setRate
public final void setRate(double permitsPerSecond) { checkArgument( permitsPerSecond > 0.0 && !Double.isNaN(permitsPerSecond), "rate must be positive"); synchronized (mutex()) { doSetRate(permitsPerSecond, stopwatch.readMicros()); } }
通過這個(gè)接口設(shè)置令牌通每秒生成令牌的數(shù)量,內(nèi)部時(shí)間通過調(diào)用SmoothRateLimiter的doSetRate來實(shí)現(xiàn)
doSetRate
@Override final void doSetRate(double permitsPerSecond, long nowMicros) { resync(nowMicros); double stableIntervalMicros = SECONDS.toMicros(1L) / permitsPerSecond; this.stableIntervalMicros = stableIntervalMicros; doSetRate(permitsPerSecond, stableIntervalMicros); }
這里先通過調(diào)用resync生成令牌以及更新下一期令牌生成時(shí)間,然后更新stableIntervalMicros,最后又調(diào)用了SmoothBursty的doSetRate
resync
/** * Updates {@code storedPermits} and {@code nextFreeTicketMicros} based on the current time. * 基于當(dāng)前時(shí)間,更新下一次請求令牌的時(shí)間,以及當(dāng)前存儲的令牌(可以理解為生成令牌) */ void resync(long nowMicros) { // if nextFreeTicket is in the past, resync to now if (nowMicros > nextFreeTicketMicros) { double newPermits = (nowMicros - nextFreeTicketMicros) / coolDownIntervalMicros(); storedPermits = min(maxPermits, storedPermits + newPermits); nextFreeTicketMicros = nowMicros; } }
根據(jù)令牌桶算法,桶中的令牌是持續(xù)生成存放的,有請求時(shí)需要先從桶中拿到令牌才能開始執(zhí)行,誰來持續(xù)生成令牌存放呢?
一種解法是,開啟一個(gè)定時(shí)任務(wù),由定時(shí)任務(wù)持續(xù)生成令牌。這樣的問題在于會極大的消耗系統(tǒng)資源,如,某接口需要分別對每個(gè)用戶做訪問頻率限制,假設(shè)系統(tǒng)中存在6W用戶,則至多需要開啟6W個(gè)定時(shí)任務(wù)來維持每個(gè)桶中的令牌數(shù),這樣的開銷是巨大的。
另一種解法則是延遲計(jì)算,如上resync函數(shù)。該函數(shù)會在每次獲取令牌之前調(diào)用,其實(shí)現(xiàn)思路為,若當(dāng)前時(shí)間晚于nextFreeTicketMicros,則計(jì)算該段時(shí)間內(nèi)可以生成多少令牌,將生成的令牌加入令牌桶中并更新數(shù)據(jù)。這樣一來,只需要在獲取令牌時(shí)計(jì)算一次即可。
SmoothBursty的doSetRate
@Override void doSetRate(double permitsPerSecond, double stableIntervalMicros) { double oldMaxPermits = this.maxPermits; maxPermits = maxBurstSeconds * permitsPerSecond; if (oldMaxPermits == Double.POSITIVE_INFINITY) { // if we don"t special-case this, we would get storedPermits == NaN, below // Double.POSITIVE_INFINITY 代表無窮啊 storedPermits = maxPermits; } else { storedPermits = (oldMaxPermits == 0.0) ? 0.0 // initial state : storedPermits * maxPermits / oldMaxPermits; } }
桶中可存放的最大令牌數(shù)由maxBurstSeconds計(jì)算而來,其含義為最大存儲maxBurstSeconds秒生成的令牌。
該參數(shù)的作用在于,可以更為靈活地控制流量。如,某些接口限制為300次/20秒,某些接口限制為50次/45秒等。也就是流量不局限于qps
在了解以上概念后,就非常容易理解RateLimiter暴露出來的接口
@CanIgnoreReturnValue public double acquire() { return acquire(1); } /** * 獲取令牌,返回阻塞的時(shí)間 **/ @CanIgnoreReturnValue public double acquire(int permits) { long microsToWait = reserve(permits); stopwatch.sleepMicrosUninterruptibly(microsToWait); return 1.0 * microsToWait / SECONDS.toMicros(1L); } final long reserve(int permits) { checkPermits(permits); synchronized (mutex()) { return reserveAndGetWaitLength(permits, stopwatch.readMicros()); } }
acquire函數(shù)主要用于獲取permits個(gè)令牌,并計(jì)算需要等待多長時(shí)間,進(jìn)而掛起等待,并將該值返回,主要通過reserve返回需要等待的時(shí)間,reserve中通過調(diào)用reserveAndGetWaitLength獲取等待時(shí)間
/** * Reserves next ticket and returns the wait time that the caller must wait for. * * @return the required wait time, never negative */ final long reserveAndGetWaitLength(int permits, long nowMicros) { long momentAvailable = reserveEarliestAvailable(permits, nowMicros); return max(momentAvailable - nowMicros, 0); }
最后調(diào)用了reserveEarliestAvailable
@Override final long reserveEarliestAvailable(int requiredPermits, long nowMicros) { resync(nowMicros); long returnValue = nextFreeTicketMicros; double storedPermitsToSpend = min(requiredPermits, this.storedPermits); double freshPermits = requiredPermits - storedPermitsToSpend; long waitMicros = storedPermitsToWaitTime(this.storedPermits, storedPermitsToSpend) + (long) (freshPermits * stableIntervalMicros); this.nextFreeTicketMicros = LongMath.saturatedAdd(nextFreeTicketMicros, waitMicros); this.storedPermits -= storedPermitsToSpend; return returnValue; }
首先通過resync生成令牌以及同步nextFreeTicketMicros時(shí)間戳,freshPermits從令牌桶中獲取令牌后還需要的令牌數(shù)量,通過storedPermitsToWaitTime計(jì)算出獲取freshPermits還需要等待的時(shí)間,在穩(wěn)定模式中,這里就是(long) (freshPermits * stableIntervalMicros) ,然后更新nextFreeTicketMicros以及storedPermits,這次獲取令牌需要的等待到的時(shí)間點(diǎn), reserveAndGetWaitLength返回需要等待的時(shí)間間隔。 從`reserveEarliestAvailable`可以看出RateLimiter的預(yù)消費(fèi)原理,以及獲取令牌的等待時(shí)間時(shí)間原理(可以解釋示例結(jié)果),再獲取令牌不足時(shí),并沒有等待到令牌全部生成,而是更新了下次獲取令牌時(shí)的nextFreeTicketMicros,從而影響的是下次獲取令牌的等待時(shí)間。 `reserve`這里返回等待時(shí)間后,`acquire`通過調(diào)用`stopwatch.sleepMicrosUninterruptibly(microsToWait);`進(jìn)行sleep操作,這里不同于Thread.sleep(), 這個(gè)函數(shù)的sleep是uninterruptibly的,內(nèi)部實(shí)現(xiàn):
public static void sleepUninterruptibly(long sleepFor, TimeUnit unit) { //sleep 阻塞線程 內(nèi)部通過Thread.sleep() boolean interrupted = false; try { long remainingNanos = unit.toNanos(sleepFor); long end = System.nanoTime() + remainingNanos; while (true) { try { // TimeUnit.sleep() treats negative timeouts just like zero. NANOSECONDS.sleep(remainingNanos); return; } catch (InterruptedException e) { interrupted = true; remainingNanos = end - System.nanoTime(); //如果被interrupt可以繼續(xù),更新sleep時(shí)間,循環(huán)繼續(xù)sleep } } } finally { if (interrupted) { Thread.currentThread().interrupt(); //如果被打斷過,sleep過后再真正中斷線程 } } }
sleep之后,`acquire`返回sleep的時(shí)間,阻塞結(jié)束,獲取到令牌。
public boolean tryAcquire(int permits) { return tryAcquire(permits, 0, MICROSECONDS); } public boolean tryAcquire() { return tryAcquire(1, 0, MICROSECONDS); } public boolean tryAcquire(int permits, long timeout, TimeUnit unit) { long timeoutMicros = max(unit.toMicros(timeout), 0); checkPermits(permits); long microsToWait; synchronized (mutex()) { long nowMicros = stopwatch.readMicros(); if (!canAcquire(nowMicros, timeoutMicros)) { return false; } else { microsToWait = reserveAndGetWaitLength(permits, nowMicros); } } stopwatch.sleepMicrosUninterruptibly(microsToWait); return true; } private boolean canAcquire(long nowMicros, long timeoutMicros) { return queryEarliestAvailable(nowMicros) - timeoutMicros <= nowMicros; } @Override final long queryEarliestAvailable(long nowMicros) { return nextFreeTicketMicros; }
tryAcquire函數(shù)可以嘗試在timeout時(shí)間內(nèi)獲取令牌,如果可以則掛起等待相應(yīng)時(shí)間并返回true,否則立即返回false
canAcquire用于判斷timeout時(shí)間內(nèi)是否可以獲取令牌,通過判斷當(dāng)前時(shí)間+超時(shí)時(shí)間是否大于nextFreeTicketMicros 來決定是否能夠拿到足夠的令牌數(shù),如果可以獲取到,則過程同acquire,線程sleep等待,如果通過canAcquire在此超時(shí)時(shí)間內(nèi)不能回去到令牌,則可以快速返回,不需要等待timeout后才知道能否獲取到令牌。
因?yàn)镾moothBursty允許一定程度的突發(fā),會有人擔(dān)心如果允許這種突發(fā),假設(shè)突然間來了很大的流量,那么系統(tǒng)很可能扛不住這種突發(fā)。因此需要一種平滑速率的限流工具,從而系統(tǒng)冷啟動后慢慢的趨于平均固定速率(即剛開始速率小一些,然后慢慢趨于我們設(shè)置的固定速率)。Guava也提供了SmoothWarmingUp來實(shí)現(xiàn)這種需求,其可以認(rèn)為是漏桶算法,但是在某些特殊場景又不太一樣。
SmoothWarmingUp創(chuàng)建方式:
public static RateLimiter create(double permitsPerSecond, long warmupPeriod, TimeUnit unit) { checkArgument(warmupPeriod >= 0, "warmupPeriod must not be negative: %s", warmupPeriod); return create( permitsPerSecond, warmupPeriod, unit, 3.0, SleepingStopwatch.createFromSystemTimer()); }
permitsPerSecond表示每秒新增的令牌數(shù),warmupPeriod表示在從冷啟動速率過渡到平均速率的時(shí)間間隔,大致原理是類似的,這里就先不分析了。
到此,Guava RateLimiter穩(wěn)定模式的實(shí)現(xiàn)原理基本已經(jīng)清楚,如發(fā)現(xiàn)文中錯(cuò)誤的地方,勞煩指正!
上述分析主要參考了:https://segmentfault.com/a/11...,再此基礎(chǔ)上做了些筆記補(bǔ)充
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/76899.html
摘要:計(jì)數(shù)限流算法無論固定窗口還是滑動窗口核心均是對請求進(jìn)行計(jì)數(shù),區(qū)別僅僅在于對于計(jì)數(shù)時(shí)間區(qū)間的處理。令牌桶限流實(shí)現(xiàn)原理令牌桶限流的實(shí)現(xiàn)原理在有詳細(xì)說明。因此由此為入口進(jìn)行分析。目前可返回的實(shí)現(xiàn)子類包括及兩種,具體不同下文詳細(xì)分析。 限流 限流一詞常用于計(jì)算機(jī)網(wǎng)絡(luò)之中,定義如下: In computer networks, rate limiting is used to control t...
摘要:自定義注解實(shí)現(xiàn)基于接口限流仔細(xì)看會發(fā)現(xiàn)上面的簡單實(shí)現(xiàn)會造成我每個(gè)接口都要寫一次限流方法代碼很冗余所以采用來使用自定義注解來實(shí)現(xiàn)。 服務(wù)限流 -- 自定義注解基于RateLimiter實(shí)現(xiàn)接口限流 令牌桶限流算法showImg(https://segmentfault.com/img/bVbgTi6?w=2420&h=1547);圖片來自網(wǎng)上 令牌桶會以一個(gè)恒定的速率向固定容量大小桶...
摘要:令牌桶算法漏桶算法漏桶漏桶的出水速度是恒定的,那么意味著如果瞬時(shí)大流量的話,將有大部分請求被丟棄掉也就是所謂的溢出。 工作中對外提供的API 接口設(shè)計(jì)都要考慮限流,如果不考慮限流,會成系統(tǒng)的連鎖反應(yīng),輕者響應(yīng)緩慢,重者系統(tǒng)宕機(jī),整個(gè)業(yè)務(wù)線崩潰,如何應(yīng)對這種情況呢,我們可以對請求進(jìn)行引流或者直接拒絕等操作,保持系統(tǒng)的可用性和穩(wěn)定性,防止因流量暴增而導(dǎo)致的系統(tǒng)運(yùn)行緩慢或宕機(jī)。 在開發(fā)高并發(fā)...
摘要:允許突發(fā)流量的平滑限流器的實(shí)現(xiàn)。實(shí)例執(zhí)行結(jié)果方法返回結(jié)果代表獲取所等待的時(shí)間。先看下示例代碼運(yùn)行效果為了預(yù)防突然暴增的流量將系統(tǒng)壓垮,很貼心的增加了預(yù)熱。 RateLimiter 類圖 showImg(https://segmentfault.com/img/bVbflrs?w=1598&h=1076); RateLimiter:作為抽象類提供一個(gè)限流器的基本的抽象方法。SmoothR...
閱讀 1830·2021-10-20 13:49
閱讀 1370·2019-08-30 15:52
閱讀 2875·2019-08-29 16:37
閱讀 1045·2019-08-29 10:55
閱讀 3079·2019-08-26 12:14
閱讀 1658·2019-08-23 17:06
閱讀 3241·2019-08-23 16:59
閱讀 2550·2019-08-23 15:42