redis應用場景:實現計數器-防止刷單 最近由於雙11要來臨,公司需要在介面請求上,做一下併發限制的處理,或者做一個防止刷單的安全攔截:比如:一個介面請求,限制每秒請求總數為200次,超過200次就等待,等下一秒,再次請求,這裡用到一個redis作為一個計數器的模式來實現。 調用redis的方法: ...
redis應用場景:實現計數器-防止刷單
最近由於雙11要來臨,公司需要在介面請求上,做一下併發限制的處理,或者做一個防止刷單的安全攔截:
比如:一個介面請求,限制每秒請求總數為200次,超過200次就等待,等下一秒,再次請求,這裡用到一個redis作為一個計數器的模式來實現。
調用redis的方法:
INCR key
將 key 中儲存的數字值增一。
如果 key 不存在,那麼 key 的值會先被初始化為 0 ,然後再執行 INCR 操作。
如果值包含錯誤的類型,或字元串類型的值不能表示為數字,那麼返回一個錯誤。
這是一個針對字元串的操作,因為 Redis 沒有專用的整數類型,所以 key 內儲存的字元串被解釋為十進位 64 位有符號整數來執行 INCR 操作。
code:redis> SET test 20
OK
redis> INCR test
(integer) 21
redis> GET test # 數字值在 Redis 中以字元串的形式保存
"21"
計數器的實現
計數器是 Redis 的原子性自增操作可實現的最直觀的模式了,它的想法相當簡單:每當某個操作發生時,向 Redis 發送一個 INCR 命令。
比如在一個 web 應用程式中,如果想知道用戶在一年中每天的點擊量,那麼只要將用戶 ID 以及相關的日期信息作為鍵,併在每次用戶點擊頁面時,執行一次自增操作即可。
比如用戶名是 peter ,點擊時間是 2012 年 3 月 22 日,那麼執行命令 INCR peter::2012.3.22 。
$redisKey = “api_name_” + $api;
$count = $this->redis->incr($redisKey);
if ($count == 1) {
//設置有效期一s
$this->redis->expire($redisKey,1);//設置一s的過期時間
}
if (count > 200) {//防止刷單的安全攔截
return false;//超過就返回false
}
//後續處理
這就簡單的實現了redis計數器的應用,另外還有以下方法:
以下幾種方式擴展這個簡單的模式:
可以通過組合使用 INCR 和 EXPIRE ,來達到只在規定的生存時間內進行計數(counting)的目的。
客戶端可以通過使用 GETSET 命令原子性地獲取計數器的當前值並將計數器清零,更多信息請參考 GETSET 命令。
使用其他自增/自減操作,比如 DECR 和 INCRBY ,用戶可以通過執行不同的操作增加或減少計數器的值,比如在游戲中的記分器就可能用到這些命令。