目錄
工作原理" >工作原理
優缺點" >優缺點
go-zero 程式碼實作" >go-zero 程式碼實作
首頁 後端開發 Golang Go中是如何實現用戶的每日限額

Go中是如何實現用戶的每日限額

Jan 10, 2022 pm 03:55 PM
go go-zero 微服務

本文由golang教學專欄為大家介紹Go 中怎麼實現使用者的每日限額,希望對需要的朋友有幫助!

Go 中實現使用者的每日限額(例如一天只能領三次福利)

如果你寫一個 bug 管理系統,使用了這個 PeriodLimit 你就可以限制每個測試者每天只能給你一個 bug。工作是不是就輕鬆很多了? :P

如今微服務架構大行其道本質原因是因為要降低系統的整體複雜度,將系統風險均攤到子系統從而最大化保證系統的穩定性,透過領域劃分拆成不同的子系統後各子系統能獨立的開發、測試、發布,研發節奏與效率能明顯提升。

但同時也帶來了問題,例如:呼叫連結過長,部署架構複雜度提升,各種中間件需要支援分散式場景。為了確保微服務的正常運行,服務治理就不可或缺了,通常包括:限流,降級,熔斷。

其中限流指的是針對介面呼叫頻率進行限制,以免超出承載上限拖垮系統。例如:

  • 電商秒殺場景

  • #API 針對不同商家限流

常用的限流演算法有:

  • 固定時間視窗限流
  • 滑動時間視窗限流
  • 漏電桶限流
  • ##令牌桶限流
本文主要講解固定時間視窗限流演算法。

從某個時間點開始每次請求過來請求數1,同時判斷目前時間視窗內請求數是否超過限制,超過限制則拒絕該請求,然後下個時間窗口開始時計數器清零等待請求。

Go中是如何實現用戶的每日限額

#實作簡單高效,特別適合用來限制例如一個用戶一天只能發10篇文章、只能發送5次短信驗證碼、只能嘗試登錄5次等場景,實際業務中此類場景非常多見。

缺點

固定時間視窗限流的缺點在於無法處理臨界區請求突發場景。

假設每 1s 限流 100 次請求,用戶在中間 500ms 時開始 1s 內發起 200 次請求,此時 200 次請求是可以全部通過的。這就和我們預期 1s 限流 100 次不合了,根源在於限流的細粒度太粗。

Go中是如何實現用戶的每日限額

core/limit/periodlimit.go

#go-zero 中使用redis 過期時間來模擬固定時間視窗。

redis lua 腳本:

-- KYES[1]:限流器key-- ARGV[1]:qos,单位时间内最多请求次数-- ARGV[2]:单位限流窗口时间-- 请求最大次数,等于p.quotalocal limit = tonumber(ARGV[1])-- 窗口即一个单位限流周期,这里用过期模拟窗口效果,等于p.permitlocal window = tonumber(ARGV[2])-- 请求次数+1,获取请求总数local current = redis.call("INCRBY",KYES[1],1)-- 如果是第一次请求,则设置过期时间并返回 成功if current == 1 then
  redis.call("expire",KYES[1],window)
  return 1-- 如果当前请求数量小于limit则返回 成功elseif current limit则返回 失败else
  return 0end
登入後複製
#固定時間視窗限流器定義

type (
  // PeriodOption defines the method to customize a PeriodLimit.
  // go中常见的option参数模式
  // 如果参数非常多,推荐使用此模式来设置参数
  PeriodOption func(l *PeriodLimit)

  // A PeriodLimit is used to limit requests during a period of time.
  // 固定时间窗口限流器
  PeriodLimit struct {
    // 窗口大小,单位s
    period     int
    // 请求上限
    quota      int
    // 存储
    limitStore *redis.Redis
    // key前缀
    keyPrefix  string
    // 线性限流,开启此选项后可以实现周期性的限流
    // 比如quota=5时,quota实际值可能会是5.4.3.2.1呈现出周期性变化
    align      bool
  }
)
登入後複製
注意一下align 參數,align= true 時請求上限將會呈現週期性的變化。

例如quota=5時實際quota可能是5.4.3.2.1呈現出週期性變化

#限流邏輯

##其實限流邏輯在上面的lua 腳本實現了,需要注意的是回傳值

0:表示錯誤,例如可能是redis 故障、過載
  • 1:允許
  • ##2:允許但是當前視窗內已到達上限,如果是跑批業務的話此時可以休眠sleep 一下等待下個視窗(作者考慮的非常細緻)
  • #3:拒絕
  • // Take requests a permit, it returns the permit state.
    // 执行限流
    // 注意一下返回值:
    // 0:表示错误,比如可能是redis故障、过载
    // 1:允许
    // 2:允许但是当前窗口内已到达上限
    // 3:拒绝
    func (h *PeriodLimit) Take(key string) (int, error) {
      // 执行lua脚本
      resp, err := h.limitStore.Eval(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
      }
    }
    登入後複製
  • 這個固定視窗限流可能用來限制例如一個用戶一天只能發送5次驗證碼短信,此時我們就需要跟中國時區對應(GMT 8),並且其實限流時間應該從零點開始,此時我們需要額外對齊(設定align 為true)。
  • // 计算过期时间也就是窗口时间大小
    // 如果align==true
    // 线性限流,开启此选项后可以实现周期性的限流
    // 比如quota=5时,quota实际值可能会是5.4.3.2.1呈现出周期性变化
    func (h *PeriodLimit) calcExpireSeconds() int {
      if h.align {
        now := time.Now()
        _, offset := now.Zone()
        unix := now.Unix() + int64(offset)
        return h.period - int(unix%int64(h.period))
      }
    
      return h.period
    }
    登入後複製
    專案位址

    github.com/zeromicro/go-zero

    歡迎使用

    go-zero

    star# 支持我們!

    以上是Go中是如何實現用戶的每日限額的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

Go WebSocket 訊息如何發送? Go WebSocket 訊息如何發送? Jun 03, 2024 pm 04:53 PM

在Go中,可以使用gorilla/websocket包發送WebSocket訊息。具體步驟:建立WebSocket連線。傳送文字訊息:呼叫WriteMessage(websocket.TextMessage,[]byte("訊息"))。發送二進位訊息:呼叫WriteMessage(websocket.BinaryMessage,[]byte{1,2,3})。

PHP框架與微服務:雲端原生部署與容器化 PHP框架與微服務:雲端原生部署與容器化 Jun 04, 2024 pm 12:48 PM

PHP框架與微服務結合的好處:可擴展性:輕鬆擴展應用程序,添加新功能或處理更多負載。靈活性:微服務獨立部署和維護,更容易進行更改和更新。高可用性:一個微服務的故障不會影響其他部分,確保更高可用性。實戰案例:使用Laravel和Kubernetes部署微服務步驟:建立Laravel專案。定義微服務控制器。建立Dockerfile。建立Kubernetes清單。部署微服務。測試微服務。

Golang 技術效能優化中如何避免記憶體洩漏? Golang 技術效能優化中如何避免記憶體洩漏? Jun 04, 2024 pm 12:27 PM

記憶體洩漏會導致Go程式記憶體不斷增加,可通過:關閉不再使用的資源,如檔案、網路連線和資料庫連線。使用弱引用防止記憶體洩漏,當物件不再被強引用時將其作為垃圾回收目標。利用go協程,協程棧記憶體會在退出時自動釋放,避免記憶體洩漏。

Golang 函數接收 map 參數時的注意事項 Golang 函數接收 map 參數時的注意事項 Jun 04, 2024 am 10:31 AM

在Go中傳遞map給函數時,預設會建立副本,對副本的修改不影響原map。如果需要修改原始map,可透過指標傳遞。空map需小心處理,因為技術上是nil指針,傳遞空map給期望非空map的函數會發生錯誤。

如何使用 Golang 的錯誤包裝器? 如何使用 Golang 的錯誤包裝器? Jun 03, 2024 pm 04:08 PM

在Golang中,錯誤包裝器允許你在原始錯誤上追加上下文訊息,從而創建新錯誤。這可用於統一不同程式庫或元件拋出的錯誤類型,簡化偵錯和錯誤處理。步驟如下:使用errors.Wrap函數將原有錯誤包裝成新錯誤。新錯誤包含原始錯誤的上下文資訊。使用fmt.Printf輸出包裝後的錯誤,提供更多上下文和可操作性。在處理不同類型的錯誤時,使用errors.Wrap函數統一錯誤類型。

Java 框架如何支援微服務的橫向擴展? Java 框架如何支援微服務的橫向擴展? Jun 04, 2024 pm 04:34 PM

Java框架支援微服務的橫向擴展,具體方式包括:SpringCloud提供Ribbon和Feign用於伺服器端和客戶端負載平衡。 NetflixOSS提供Eureka和Zuul,實現服務發現、負載平衡和故障轉移。 Kubernetes透過自動擴展、健康檢查和自動重新啟動簡化了橫向擴展。

如何在 Go 中創建優先級 Goroutine? 如何在 Go 中創建優先級 Goroutine? Jun 04, 2024 pm 12:41 PM

在Go語言中建立優先權Goroutine有兩步驟:註冊自訂Goroutine建立函數(步驟1)並指定優先權值(步驟2)。這樣,您可以建立不同優先順序的Goroutine,優化資源分配並提高執行效率。

使用 Golang 微服務框架建立分散式系統 使用 Golang 微服務框架建立分散式系統 Jun 05, 2024 pm 06:36 PM

使用Golang微服務框架建立分散式系統:安裝Golang、選擇微服務框架(如Gin)建立Gin微服務,新增端點部署微服務,建置並執行應用程式建立訂單和庫存微服務,使用端點處理訂單和庫存使用Kafka等訊息傳遞系統連接微服務使用sarama庫生產和消費訂單訊息

See all articles