Gin框架限流實現(xiàn)示例
什么是限流
限流是指通過一定的算法,對接口的請求進行限制,防止并發(fā)量過大,導致系統(tǒng)癱瘓或響應變慢的情況出現(xiàn)。
為什么要進行限流
在高并發(fā)的場景下,如果不進行限流,系統(tǒng)可能會因為過多的請求而崩潰。限流可以保護系統(tǒng)免于被流量打崩,從而保證系統(tǒng)的可用性和穩(wěn)定性。
Gin框架的限流實現(xiàn)
Gin 是一個基于 Go 語言的 web 框架,它提供了很多方便的中間件,可以方便地實現(xiàn)限流。
以下是一個基于 Gin 實現(xiàn)的令牌桶限流的例子:
定義令牌桶結(jié)構(gòu)體
type TokenBucket struct { capacity int64 // 桶的容量 rate float64 // 令牌放入速率 tokens float64 // 當前令牌數(shù)量 lastToken time.Time // 上一次放令牌的時間 mtx sync.Mutex // 互斥鎖 }實現(xiàn)令牌桶算法
func (tb *TokenBucket) Allow() bool { tb.mtx.Lock() defer tb.mtx.Unlock() now := time.Now() // 計算需要放的令牌數(shù)量 tb.tokens = tb.tokens + tb.rate*now.Sub(tb.lastToken).Seconds() if tb.tokens > float64(tb.capacity) { tb.tokens = float64(tb.capacity) } // 判斷是否允許請求 if tb.tokens >= 1 { tb.tokens-- tb.lastToken = now return true } else { return false } }使用中間件進行限流
func LimitHandler(maxConn int) gin.HandlerFunc { tb := &TokenBucket{ capacity: maxConn, rate: 1.0, tokens: 0, lastToken: time.Now(), } return func(c *gin.Context) { if !tb.Allow() { c.String(503, "Too many request") c.Abort() return } c.Next() } }在路由中使用中間件
r := gin.Default() // 在路由中使用中間件 r.Use(LimitHandler(100)) r.GET("/", func(c *gin.Context) { c.String(200, "Hello, World!") }) r.Run(":8080")
以上代碼實現(xiàn)了一個簡單的令牌桶限流中間件,可以限制最大并發(fā)連接數(shù)為 100。如果超過了這個連接數(shù),將會返回 503 狀態(tài)碼。
測試
瀏覽器地址欄輸入http://localhost:8080/, 然后瘋狂刷新即可.
- 測試截圖

總結(jié)
總的來說,使用 Gin 框架進行限流是一個方便有效的方法,可以提高系統(tǒng)的可用性和穩(wěn)定性,避免因為過多的請求導致系統(tǒng)崩潰的問題。利用令牌桶算法實現(xiàn)限流可以很好地控制請求的并發(fā)量,可以通過控制桶容量和放入速率等參數(shù)進行調(diào)節(jié)和優(yōu)化。在使用中間件進行限流時,應該根據(jù)實際應用場景和需求調(diào)節(jié)限流參數(shù),祝您的應用愉快運行!
完整代碼
package main
import (
"github.com/gin-gonic/gin"
"sync"
"time"
)
type TokenBucket struct {
capacity int64 // 桶的容量
rate float64 // 令牌放入速率
tokens float64 // 當前令牌數(shù)量
lastToken time.Time // 上一次放令牌的時間
mtx sync.Mutex // 互斥鎖
}
func (tb *TokenBucket) Allow() bool {
tb.mtx.Lock()
defer tb.mtx.Unlock()
now := time.Now()
// 計算需要放的令牌數(shù)量
tb.tokens = tb.tokens + tb.rate*now.Sub(tb.lastToken).Seconds()
if tb.tokens > float64(tb.capacity) {
tb.tokens = float64(tb.capacity)
}
// 判斷是否允許請求
if tb.tokens >= 1 {
tb.tokens--
tb.lastToken = now
return true
} else {
return false
}
}
func LimitHandler(maxConn int64) gin.HandlerFunc {
tb := &TokenBucket{
capacity: maxConn,
rate: 1.0,
tokens: 0,
lastToken: time.Now(),
}
return func(c *gin.Context) {
if !tb.Allow() {
c.String(503, "Too many request")
c.Abort()
return
}
c.Next()
}
}
func main() {
r := gin.Default()
// 在路由中使用中間件
r.Use(LimitHandler(100))
r.GET("/", func(c *gin.Context) {
c.String(200, "Hello, World!")
})
r.Run(":8080")
}到此這篇關于Gin框架限流實現(xiàn)示例的文章就介紹到這了,更多相關Gin框架限流內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
golang gin 監(jiān)聽rabbitmq隊列無限消費的案例代碼
這篇文章主要介紹了golang gin 監(jiān)聽rabbitmq隊列無限消費,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-12-12
golang調(diào)試bug及性能監(jiān)控方式實踐總結(jié)
這篇文章主要為大家介紹了golang調(diào)試bug及性能監(jiān)控方式實踐是總結(jié),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-05-05

