這篇文章將為大家詳細(xì)講解有關(guān)php基于redis的list型數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)ip限流操作的示例,小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。
成都創(chuàng)新互聯(lián)是一家專注于網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)與策劃設(shè)計(jì),信州網(wǎng)站建設(shè)哪家好?成都創(chuàng)新互聯(lián)做網(wǎng)站,專注于網(wǎng)站建設(shè)10余年,網(wǎng)設(shè)計(jì)領(lǐng)域的專業(yè)建站公司;建站業(yè)務(wù)涵蓋:信州等地區(qū)。信州做網(wǎng)站價(jià)格咨詢:18982081108
在日常的業(yè)務(wù)功能開發(fā)中,如果要 限制任意一個(gè)ip在連續(xù)的某一段時(shí)間內(nèi),只能訪問某個(gè)接口一定的次數(shù),需要如何實(shí)現(xiàn)呢?
這種功能需求通常是用來應(yīng)對(duì)防止腳本惡意刷接口的情況,目前網(wǎng)上已經(jīng)有很多比較完善的限流方案。對(duì)于一般的站點(diǎn)來講,可以借助redis的鏈表型數(shù)據(jù)結(jié)構(gòu)來實(shí)現(xiàn)ip限流功能。
舉個(gè)例子——
假如我們需要實(shí)現(xiàn),對(duì)于接口A,限制任意IP在每一段連續(xù)的5秒內(nèi),最多允許3次訪問,超過3次則返回報(bào)錯(cuò)。
對(duì)于上圖來講,在08秒的時(shí)候,最近的5秒內(nèi)已經(jīng)發(fā)起了4次請(qǐng)求,已經(jīng)達(dá)到最大次數(shù)限制,所以此時(shí)訪問會(huì)受限。
采用PHP來實(shí)現(xiàn)的話,具體的邏輯代碼如下——
/** * 檢查隊(duì)列的長度是否到達(dá)設(shè)定的閾值,已到達(dá)則返回false,未到達(dá)則將當(dāng)前時(shí)間戳推入隊(duì)列最末端,同時(shí)刷新隊(duì)列整體的緩存時(shí)間 * @param $key 隊(duì)列緩存的key * @param $expire 隊(duì)列緩存過期時(shí)間,例如上面例子中的5秒 * @param $limit 隊(duì)列長度閾值,如上面例子中的3次 * @return bool */public function checkLimit($key, $expire, $limit){ $length = $this->refreshList($key, $expire); if ($length < $limit) { // 未到達(dá)訪問限制,將當(dāng)前時(shí)間戳推入到list的最后邊,同時(shí)把整個(gè)key的過期時(shí)間重新更新 $this->rPush($key, time()); $this->expire($key, intval($limit)); return true; } return false;}/** * 刷新隊(duì)列,過濾掉已經(jīng)不在有效時(shí)間內(nèi)的值,返回最新隊(duì)列的長度 * @param $key string 自定義的緩存key * @param $expire 隊(duì)列緩存過期時(shí)間,例如上面例子中的5秒 * @return bool|int */public function refreshList ($key, $expire) { if ($this->has($key)) { do { // 對(duì)于已存在數(shù)據(jù)的list,要先從前往后把已經(jīng)過期的數(shù)據(jù)彈出 $oldest_value = $this->lPop($key); } while ($oldest_value && time() - $oldest_value > $expire); // 把最后彈出的數(shù)據(jù)重新塞回list的最前邊 $oldest_value && $this->lPush($key, $oldest_value); return $this->lLen($key); } return 0;}
其中用到的lPop,lPush,lLen,rPush等方法,都是封裝了redis拓展之后,操作鏈表型數(shù)據(jù)結(jié)構(gòu)的一些方法,參數(shù)跟返回值都與原生方法保持一致。
其實(shí)后來網(wǎng)上查了之后才知道,redis處理這種場景,更多的是直接用zset這種有序集來實(shí)現(xiàn),邏輯也是基本一致,就是存當(dāng)前時(shí)間戳,然后用滑動(dòng)窗口的算法思想,判斷當(dāng)前窗口內(nèi)的值長度是否已經(jīng)超過限制。
關(guān)于“php基于redis的list型數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)ip限流操作的示例”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,使各位可以學(xué)到更多知識(shí),如果覺得文章不錯(cuò),請(qǐng)把它分享出去讓更多的人看到。