国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 數據庫 > Redis > 正文

在Redis數據庫中實現分布式速率限制的方法

2020-03-17 12:42:07
字體:
來源:轉載
供稿:網友

這篇文章主要介紹了在Redis數據庫中實現分布式速率限制的方法,文中展示了一個用Python編寫的應用示例,需要的朋友可以參考下

問題

在許多應用中,對昂貴的資源的訪問必須加以限制,此時速率限制是必不可少的。許多現代網絡應用程序在多個進程和服務器上運行,狀態需要被共享。一個理想的解決方案應該是高效、 快捷的,而不是依賴于被綁定到特定客戶端的單個應用程序服務器(由于負載平衡) 或本身持有任何狀態。

解決方案

實現這一目標的一個簡單有效的方法就是使用 Redis, 它有很多有用的數據結構和功能, 盡管實現速率限制只需要2個功能用: 一、在某個具體的鍵值上遞增一個整數,二、給這個鍵值設置過期時間。

因為redis 有個單一的事件循環系統 (每個人每次在同一個時間只能執行一個操作),這是個原子操作, 也就是說無論有多少個客戶端同時交互操作,對于同一個鍵值總有一個確定的數值。

這在對同一個資源進行多個速率限制的情況下通常是有利的, 因為這允許少量的破裂,以及更長的期限限制。例如每秒鐘請求3次,沒分鐘請求20次。因為每個限制都是相對獨立的,這就需要與其它限制分開進行單獨的遞增。

因為速率限制通常用在響應時間比較重要的資源(比如網頁應用),所以盡量縮短速率限制的使用時間是非常有必要的。redis的最基本的應用就是發出命令,等待響應,然后發出另一個命令,如此往復。 這個花費是昂貴的,因為需要通過網絡在應用程序和redis服務器之間多次往返。由于在這個用例中,沒有命令依賴其它命令的執行結果,這使得redis的一個叫做流水線技術的使用成為可能。這就是客戶端緩存所有redis請求,然后把這寫請求發送給redis,redis一次性返回所有的結果。

Redis不會維護客戶端需要的限制的,因為redis會根據客戶端設置的過期時間刪除舊的記數。這消除了客戶端統籌協調的需要,和刪除競爭條件的可能性。

The Code

 

 
  1. import redis 
  2. import time 
  3.  
  4. def rate_limit_check(r, key, limits): 
  5. period_lengths = [_[0] for _ in sorted(limits.items())] 
  6. period_limits = [_[1] for _ in sorted(limits.items())] 
  7. pipe = r.pipeline() 
  8. for period_length in period_lengths: 
  9. current_period = int(time.time() / period_length) 
  10. redis_key = 'rate_limit:{key}:{period_length}:{current_period}'.format(key=key, period_length=period_length, current_period=current_period) 
  11. pipe.incr(redis_key).expire(redis_key, period_length*3) 
  12. return not any(hits > period_limit for period_limit, hits in zip(period_limits, pipe.execute()[::2])) 
  13.  
  14. if __name__ == '__main__'
  15. r = redis.Redis() 
  16. print rate_limit_check(r, '127.0.0.1', {1: 3, 60: 20}) 

{1: 3, 60: 20} 意味著每秒鐘3次的命中率是允許的,在任何限制下,都允許20次的命中。'127.0.0.1'在這里用作鍵值,盡管在真實的情況下,可能作為IP地址。更高級的用例將有一個全應用程序的速率限制,鍵值只有客戶端的IP地址,以及一個為昂貴的終結點設置的特定終結點限制,這將用到客戶端的IP地址和終結點,例如127.0.0.1+/login/。這些限制可以獨立地設置。

 

 
  1. return rate_limit_check(r, '127.0.0.1', {1: 3, 60: 20}) and rate_limit_check(r, '127.0.0.1+/login/', {1: 2, 60: 5}) 

這是一個用Python寫的例子,它可以簡單地移植到任何語言,只要這門語言包含Redis客戶端庫。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 景德镇市| 浙江省| 宕昌县| 昭平县| 南部县| 交城县| 三门峡市| 长乐市| 西丰县| 泸州市| 昭平县| 海淀区| 兰考县| 宁津县| 剑川县| 安徽省| 宁阳县| 梓潼县| 东明县| 九江市| 深泽县| 得荣县| 平度市| 永胜县| 泰宁县| 永和县| 道孚县| 西贡区| 安溪县| 中西区| 聂荣县| 广东省| 顺平县| 额济纳旗| 沾化县| 颍上县| 江阴市| 昌平区| 通化县| 南江县| 盖州市|