欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

redis限流的實際應(yīng)用

 更新時間:2021年04月23日 10:46:43   作者:程序員養(yǎng)成日記  
這篇文章主要介紹了redis限流的實際應(yīng)用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

為什么要做限流

首先讓我們先看一看系統(tǒng)架構(gòu)設(shè)計中,為什么要做“限流”。

旅游景點通常都會有最大的接待量,不可能無限制的放游客進入,比如故宮每天只賣八萬張票,超過八萬的游客,無法買票進入,因為如果超過八萬人,景點的工作人員可能就忙不過來,過于擁擠的景點也會影響游客的體驗和心情,并且還會有安全隱患;「只賣N張票,這就是一種限流的手段」。

軟件架構(gòu)中的服務(wù)限流也是類似,也是當(dāng)系統(tǒng)資源不夠的時候,已經(jīng)不足以應(yīng)對大量的請求,為了保證服務(wù)還能夠正常運行,那么按照規(guī)則,「系統(tǒng)會把多余的請求直接拒絕掉,以達到限流的效果」;

不知道大家注意過沒有,比如雙11,剛過12點有些顧客的網(wǎng)頁或APP會顯示下單失敗的提示,有些就是被限流掉了。

常見的限流算法

計數(shù)法

顧名思義就是來一個,記錄一個,比如我1分鐘只能處理1000個請求,那么我們就可以設(shè)置一個計數(shù)器,來一個請求就incr+1,當(dāng)1分鐘之內(nèi)的數(shù)量大于等于1000之后不處理了即可,偽代碼如下

$redis = new Redis();
$redis->connect('127.0.0.1', 6379);
$rate_limit = 1000;  //限制個數(shù)
$rate_seconds = 60;  //限制時間
$redis_key = "redis_limit";
$count = $redis->get($redis_key);
if ($count >= $rate_limit){  //判斷60秒內(nèi)請求個數(shù)是否已經(jīng)達到上限
    //直接返回,不處理請求
    return
}
$redis->incr($redis_key, 1);//請求計數(shù)
$redis->expire($redis, $rate_seconds); //設(shè)置過期時間 60s
//to do  業(yè)務(wù)邏輯處理.......

這種計數(shù)方式比較簡單快捷,但是有很大的缺點,因為請求的訪問不一定是很平穩(wěn)的,如果0:59過來了1000個請求,1:01已經(jīng)是下一個窗口,又過來了1000個請求,但實際上三秒內(nèi)來了2000個請求,已經(jīng)超過我們的限流上限了。所以這種方法是不推薦的。

滑動窗口算法

還拿上面的例子,一分鐘分6份,每份10秒;每過10秒鐘,我們的時間窗口就會往右滑動一格,每個格子都有獨立的計數(shù)器,我們每次都計算時間窗口內(nèi)的數(shù)量,可以解決計數(shù)器法中的問題,而且當(dāng)滑動窗口的格子越多,那么限流的統(tǒng)計就會越精確。具體可以參考下圖,看圖比較清晰

偽代碼實現(xiàn)如下

function api_limit($scene,  $period, $maxCount){
    $redis = new Redis();
    $redis->connect('127.0.0.1', 6379);
    $key = sprintf('hist:%s', $scene); //限流場景唯一標(biāo)識
    $now = msectime();   // 毫秒時間戳,這樣更精確
    $pipe=$redis->multi(Redis::PIPELINE); //使用管道提升性能
    $pipe->zadd($key, $now, $now); //value 和 score 都使用毫秒時間戳
    $pipe->zremrangebyscore($key, 0, $now - $period); //移除時間窗口之前的行為記錄,剩下的都是時間窗口內(nèi)的
    $pipe->zcard($key);  //獲取窗口內(nèi)的行為數(shù)量
    $pipe->expire($key, $period/1000 + 1);  //多加一秒過期時間
    $replies = $pipe->exec();
    return $replies[2] <= $maxCount;  //$replies[2]為zcard返回的個數(shù)  如果zcard結(jié)果大于maxCount,則不處理結(jié)果
}

for ($i=0; $i<20; $i++){  //測試限流是否實現(xiàn)代碼
    var_dump(isActionAllowed("uniq_scene", 60*1000, 5)); //執(zhí)行可以發(fā)現(xiàn)只有前5次是通過的
}

//返回當(dāng)前的毫秒時間戳
function msectime() {
    list($msec, $sec) = explode(' ', microtime());
    $msectime = (float)sprintf('%.0f', (floatval($msec) + floatval($sec)) * 1000);
    return $msectime;
 }

這段代碼還是略顯復(fù)雜,需要讀者花一定的時間好好啃。它的整體思路就是:每一個行為到來時,都維護一次時間窗口。將時間窗口外的記錄全部清理掉,只保留窗口內(nèi)的記錄。

因為這幾個連續(xù)的 Redis 操作都是針對同一個 key 的,使用 pipeline 可以顯著提升Redis 存取效率。「但這種方案也有缺點,因為它要記錄時間窗口內(nèi)所有的行為記錄,如果這個量很大,比如限定 60s 內(nèi)操作不得超過 100w 次這樣的參數(shù),它是不適合做這樣的限流的,因為會消耗大量的存儲空間」。

后面還有漏桶算法和令牌桶算法,由于各自的實現(xiàn)比較復(fù)雜,所以準(zhǔn)備各自新開一篇文章單獨描述

到此這篇關(guān)于redis限流的實際應(yīng)用的文章就介紹到這了,更多相關(guān)redis限流內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • caffeine_redis自定義二級緩存

    caffeine_redis自定義二級緩存

    這篇文章詳細(xì)介紹了caffeine_redis 自定義二級緩存,文中有相關(guān)的背景前提與出現(xiàn)的問題,感興趣的同學(xué)可以參考一下
    2023-04-04
  • Redis報錯:Could not create server TCP listening socket 127.0.0.1:6379: bind:解決方法

    Redis報錯:Could not create server TCP 

    這篇文章主要介紹了Redis報錯:Could not create server TCP listening socket 127.0.0.1:6379: bind:解決方法,是安裝與啟動Redis過程中比較常見的問題,需要的朋友可以參考下
    2023-06-06
  • Redis實現(xiàn)短信驗證碼登錄的示例代碼

    Redis實現(xiàn)短信驗證碼登錄的示例代碼

    本文主要介紹了基于Redis如何實現(xiàn)短信驗證碼登錄功能,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • Redis?key-value亂碼的解決

    Redis?key-value亂碼的解決

    本文主要介紹了Redis?key-value亂碼的解決,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • Redis實現(xiàn)唯一計數(shù)的3種方法分享

    Redis實現(xiàn)唯一計數(shù)的3種方法分享

    這篇文章主要介紹了Redis實現(xiàn)唯一計數(shù)的3種方法分享,本文講解了基于SET、基于 bit、基于 HyperLogLog三種方法,需要的朋友可以參考下
    2015-03-03
  • 詳解Redis高效恢復(fù)策略內(nèi)存快照與AOF

    詳解Redis高效恢復(fù)策略內(nèi)存快照與AOF

    這篇文章主要為大家介紹了Redis高效恢復(fù)策略內(nèi)存快照與AOF及對比詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-12-12
  • Redis使用元素刪除的布隆過濾器來解決緩存穿透問題

    Redis使用元素刪除的布隆過濾器來解決緩存穿透問題

    本文主要介紹了Redis使用元素刪除的布隆過濾器來解決緩存穿透問題,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • SpringBoot集成redis錯誤問題及解決方法

    SpringBoot集成redis錯誤問題及解決方法

    這篇文章主要介紹了SpringBoot集成redis錯誤問題,本文給大家分享完美解決方法,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-02-02
  • Redis做數(shù)據(jù)持久化的解決方案及底層原理

    Redis做數(shù)據(jù)持久化的解決方案及底層原理

    Redis有兩種方式來實現(xiàn)數(shù)據(jù)的持久化,分別是RDB(Redis Database)和AOF(Append Only File),今天通過本文給大家聊一聊Redis做數(shù)據(jù)持久化的解決方案及底層原理,感興趣的朋友一起看看吧
    2021-07-07
  • Redis?存儲對象信息用?Hash?和String的區(qū)別

    Redis?存儲對象信息用?Hash?和String的區(qū)別

    這篇文章主要介紹了Redis存儲對象信息用Hash和String的區(qū)別,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-09-09

最新評論