go-zero 是如何實現計數器限流的?

来源:https://www.cnblogs.com/alwaysbeta/archive/2023/08/10/17618200.html
-Advertisement-
Play Games

**原文鏈接:** [如何實現計數器限流?](https://mp.weixin.qq.com/s/CTemkZ2aKPCPTuQiDJri0Q) 上一篇文章 [go-zero 是如何做路由管理的?](https://mp.weixin.qq.com/s/uTJ1En-BXiLvH45xx0eFsA ...


原文鏈接: 如何實現計數器限流?

上一篇文章 go-zero 是如何做路由管理的? 介紹了路由管理,這篇文章來說說限流,主要介紹計數器限流演算法,具體的代碼實現,我們還是來分析微服務框架 go-zero 的源碼。

在微服務架構中,一個服務可能需要頻繁地與其他服務交互,而過多的請求可能導致性能下降或系統崩潰。為了確保系統的穩定性和高可用性,限流演算法應運而生。

限流演算法允許在給定時間段內,對服務的請求流量進行控制和調整,以防止資源耗盡和服務過載。

計數器限流演算法主要有兩種實現方式,分別是:

  1. 固定視窗計數器
  2. 滑動視窗計數器

下麵分別來介紹。

固定視窗計數器

演算法概念如下:

  • 將時間劃分為多個視窗;
  • 在每個視窗內每有一次請求就將計數器加一;
  • 如果計數器超過了限制數量,則本視窗內所有的請求都被丟棄當時間到達下一個視窗時,計數器重置。

固定視窗計數器是最為簡單的演算法,但這個演算法有時會讓通過請求量允許為限制的兩倍。

考慮如下情況:限制 1 秒內最多通過 5 個請求,在第一個視窗的最後半秒內通過了 5 個請求,第二個視窗的前半秒內又通過了 5 個請求。這樣看來就是在 1 秒內通過了 10 個請求。

滑動視窗計數器

演算法概念如下:

  • 將時間劃分為多個區間;
  • 在每個區間內每有一次請求就將計數器加一維持一個時間視窗,占據多個區間;
  • 每經過一個區間的時間,則拋棄最老的一個區間,並納入最新的一個區間;
  • 如果當前視窗內區間的請求計數總和超過了限制數量,則本視窗內所有的請求都被丟棄。

滑動視窗計數器是通過將視窗再細分,並且按照時間滑動,這種演算法避免了固定視窗計數器帶來的雙倍突發請求,但時間區間的精度越高,演算法所需的空間容量就越大。

go-zero 實現

go-zero 實現的是固定視窗的方式,計算一段時間內對同一個資源的訪問次數,如果超過指定的 limit,則拒絕訪問。當然如果在一段時間內訪問不同的資源,每一個資源訪問量都不超過 limit,此種情況是不會拒絕的。

而在一個分散式系統中,存在多個微服務提供服務。所以當瞬間的流量同時訪問同一個資源,如何讓計數器在分散式系統中正常計數?

這裡要解決的一個主要問題就是計算的原子性,保證多個計算都能得到正確結果。

通過以下兩個方面來解決:

  • 使用 redis 的 incrby 做資源訪問計數
  • 採用 lua script 做整個視窗計算,保證計算的原子性

接下來先看一下 lua script 的源碼:

// core/limit/periodlimit.go

const periodScript = `local limit = tonumber(ARGV[1])
local window = tonumber(ARGV[2])
local current = redis.call("INCRBY", KEYS[1], 1)
if current == 1 then
    redis.call("expire", KEYS[1], window)
end
if current < limit then
    return 1
elseif current == limit then
    return 2
else
    return 0
end`

主要就是使用 INCRBY 命令來實現,第一次請求需要給 key 加上一個過期時間,到達過期時間之後,key 過期被清楚,重新計數。

限流器初始化:

type (
    // PeriodOption defines the method to customize a PeriodLimit.
    PeriodOption func(l *PeriodLimit)

    // A PeriodLimit is used to limit requests during a period of time.
    PeriodLimit struct {
        period     int  // 視窗大小,單位 s
        quota      int  // 請求上限
        limitStore *redis.Redis
        keyPrefix  string   // key 首碼
        align      bool
    }
)

// NewPeriodLimit returns a PeriodLimit with given parameters.
func NewPeriodLimit(period, quota int, limitStore *redis.Redis, keyPrefix string,
    opts ...PeriodOption) *PeriodLimit {
    limiter := &PeriodLimit{
        period:     period,
        quota:      quota,
        limitStore: limitStore,
        keyPrefix:  keyPrefix,
    }

    for _, opt := range opts {
        opt(limiter)
    }

    return limiter
}

調用限流:

// key 就是需要被限制的資源標識
func (h *PeriodLimit) Take(key string) (int, error) {
    return h.TakeCtx(context.Background(), key)
}

// TakeCtx requests a permit with context, it returns the permit state.
func (h *PeriodLimit) TakeCtx(ctx context.Context, key string) (int, error) {
    resp, err := h.limitStore.EvalCtx(ctx, periodScript, []string{h.keyPrefix + key}, []string{
        strconv.Itoa(h.quota),
        strconv.Itoa(h.calcExpireSeconds()),
    })
    if err != nil {
        return Unknown, err
    }

    code, ok := resp.(int64)
    if !ok {
        return Unknown, ErrUnknownCode
    }

    switch code {
    case internalOverQuota: // 超過上限
        return OverQuota, nil
    case internalAllowed:   // 未超過,允許訪問
        return Allowed, nil
    case internalHitQuota:  // 正好達到限流上限
        return HitQuota, nil
    default:
        return Unknown, ErrUnknownCode
    }
}

上文已經介紹了,固定時間視窗會有臨界突發問題,並不是那麼嚴謹,下篇文章我們來介紹令牌桶限流。

以上就是本文的全部內容,如果覺得還不錯的話歡迎點贊轉發關註,感謝支持。


參考文章:

推薦閱讀:


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • # CSS 選擇器權重 聲明:本文參考來源於[MDN CSS 優先順序](https://developer.mozilla.org/zh-CN/docs/Web/CSS/Specificity)(80%)和B站尚矽谷的[課程](https://m.bilibili.com/video/BV1p84y ...
  • 當我們疲於開發一個接一個的需求時,很容易忘記去關註網站的性能,到了某一個節點,猛地發現,隨著越來越多代碼的堆積,網站變得越來越慢。本文就是從這樣的一個背景出發,著手優化網站的前端性能,並總結出一套開發習慣,讓我們在日常開發時,也保持高性能,而不是又一次回過頭來優化性能。 ...
  • 隨著大屏幕技術的發展,大屏數字滾動效果在各種應用場景中越來越常見,例如數字展示、統計數據展示等。Vue3 作為一種流行的前端開發框架,提供了強大的工具和便捷的開發方式,非常適合實現大屏數字滾動效果。 本篇博文將介紹如何使用 Vue3 來實現大屏數字滾動效果。在實現過程中,我們可以使用調試工具進行測... ...
  • 通過編碼實戰瞭解quarkus攔截器的另一個高級特性:禁用類級別攔截器,這樣可以避免類級別和方法級別攔截器的疊加衝突 ...
  • # docker-compose ## compose安裝 ### Compose 簡介 Compose 是用於定義和運行多容器 Docker 應用程式的工具。通過 Compose,您可以使用 YML 文件來配置應用程式需要的所有服務。然後,使用一個命令,就可以從 YML 文件配置中創建並啟動所有服 ...
  • Spring可能成為您的所有企業應用程式的一站式商店。但是,Spring是模塊化的,允許您挑選適用於您的模塊,而無需引入其他模塊。下麵的部分提供了Spring Framework中所有可用模塊的詳細信息。 Spring Framework提供了大約20個模塊,可以根據應用程式要求使用。 ## 核心容 ...
  • 由於垃圾收集演算法的實現涉及大量的程式細節,而且各個平臺的虛擬機操作記憶體的方法又各不相同,因此本節不打算過多地討論演算法的實現,只是介紹幾種演算法的思想及其發展過程。 垃圾收集演算法概要 1、 標記-清除演算法 標記-清除演算法最基礎的收集演算法是“標記-清除”(Mark-Sweep)演算法,演算法分為“標記”和“清 ...
  • **** # 1.內容 | | 解釋 | | | | | // 內容 | 單行註釋 | | /* 內容 */ | 多行註釋 | | /*** 內容 */ | 文檔註釋 | # 2.多行註釋 與 文檔註釋的區別 多行註釋: ![img](https://img2023.cnblogs.com/blog ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...