成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

如何實現計數器限流?

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

上一篇文章 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
    }
}

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

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

參考文章:

責任編輯:武曉燕 來源: AlwaysBeta
相關推薦

2023-09-26 08:39:28

限流方式滑動窗口計數器

2009-12-01 15:01:07

PHP生成訪問計數器

2009-11-06 16:59:26

WCF性能計數器

2023-07-28 08:15:27

PC程序計數器

2025-05-23 10:10:00

限流算法系統Go

2015-03-24 13:46:29

C++多線程計數器特性實現

2009-10-29 11:47:15

ADO.NET計數器b

2010-02-22 16:34:17

WCF性能計數器

2022-08-29 19:51:58

CSS計數器

2024-03-04 08:53:50

海量數據計數器存儲

2009-11-25 15:07:39

PHP添加計數器

2009-06-11 16:27:18

科學型Java計數器

2009-12-22 13:25:58

WCF性能計數器內存

2011-03-31 16:03:20

cacti性能計數器

2011-04-06 10:03:08

Cacti遠程監控

2009-12-22 13:16:53

WCF性能計數器

2022-04-08 07:22:15

分布式計數器系統設計

2023-02-20 08:08:48

限流算法計數器算法令牌桶算法

2010-01-04 16:02:32

ADO.NET計數器

2009-12-22 13:34:07

WCF性能計數器級別
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91精品国产日韩91久久久久久 | 精品亚洲一区二区三区四区五区 | 久久久久久久一区二区三区 | 国产一区视频在线 | 久久精品久久久久久 | 成人在线观看中文字幕 | 日本欧美国产在线 | 超碰在线播 | 午夜精品一区二区三区在线观看 | 国产精品一区二区三区在线 | 欧美三级在线 | 国产情侣久久 | 国产精品美女久久久久 | 亚洲精品1区 | 中文字幕日韩一区 | 久久久久国产 | 成人h动漫精品一区二区器材 | 九九色综合 | 99精品99 | 精品国产乱码久久久久久蜜臀 | 成人三级在线观看 | 国产成人精品免高潮在线观看 | 狠狠操狠狠搞 | 九九久久99 | 91成人在线 | 我爱操| 一区二区三区四区国产 | 久久精品免费 | 国产成人91视频 | 欧美9999 | 日韩精品一区二区三区高清免费 | 一区二区三区久久久 | 涩涩视频在线播放 | 九九免费观看视频 | 久久免费精彩视频 | 在线男人天堂 | 男女久久久 | 亚州综合一区 | a级毛片免费高清视频 | 在线观看h视频 | 国产欧美视频一区二区 |