php redis實作令牌桶演算法程式碼:
<?phpnamespace Api\Lib;/** * 限流控制 */class RateLimit{ private $minNum = 60; //单个用户每分访问数 private $dayNum = 10000; //单个用户每天总的访问量 public function minLimit($uid) { $minNumKey = $uid . '_minNum'; $dayNumKey = $uid . '_dayNum'; $resMin = $this->getRedis($minNumKey, $this->minNum, 60); $resDay = $this->getRedis($minNumKey, $this->minNum, 86400); if (!$resMin['status'] || !$resDay['status']) { exit($resMin['msg'] . $resDay['msg']); } } public function getRedis($key, $initNum, $expire) { $nowtime = time(); $result = ['status' => true, 'msg' => '']; $redisObj = $this->di->get('redis'); $redis->watch($key); $limitVal = $redis->get($key); if ($limitVal) { $limitVal = json_decode($limitVal, true); $newNum = min($initNum, ($limitVal['num'] - 1) + (($initNum / $expire) * ($nowtime - $limitVal['time']))); if ($newNum > 0) { $redisVal = json_encode(['num' => $newNum, 'time' => time()]); } else { return ['status' => false, 'msg' => '当前时刻令牌消耗完!']; } } else { $redisVal = json_encode(['num' => $initNum, 'time' => time()]); } $redis->multi(); $redis->set($key, $redisVal); $rob_result = $redis->exec(); if (!$rob_result) { $result = ['status' => false, 'msg' => '访问频次过多!']; } return $result; }}
程式碼重點:
1、先定義規則
#單一用戶每分鐘造訪次數($minNum),單一使用者每天總的造訪次數($dayNum),介面總的造訪次數等不同的規則。
2、計算速率
此程式碼範例以秒為最小的時間單位,速率=造訪次數/時間($initNum / $expire)
3、每次訪問後補充的令牌個數計算方式
取得上次造訪的時間即上次存入令牌的時間,計算當前時刻與上次造訪的時間差乘以速率就是此次需要補充的令牌個數,注意補充令牌後總的令牌個數不能大於初始化的令牌個數,以補充數和初始化數的最小值為準。
4、程式流程
第一次造訪時初始化令牌個數($minNum),存入Redis同時將當前的時間戳記存入以便計算下次需要補充的命令牌個數。
第二次訪問時獲取剩餘的令牌個數,並添加本次應該補充的令牌個數,補充後如何令牌數>0則當前訪問是有效的可以訪問,否則令牌使用完畢不可訪問。先補充令牌再判斷令牌是否>0的原因是由於還有速率這個概念即如果上次剩餘的令牌為0但是本次應該補充的令牌>1那麼本次依然可以存取。
5、針對並發的處理
使用Redis的樂觀鎖定機制。
更多相關知識,請追蹤 PHP中文網! !
以上是令牌桶演算法如何使用php實現的詳細內容。更多資訊請關注PHP中文網其他相關文章!