Gin框架限流實(shí)現(xiàn)示例
什么是限流
限流是指通過一定的算法,對接口的請求進(jìn)行限制,防止并發(fā)量過大,導(dǎo)致系統(tǒng)癱瘓或響應(yīng)變慢的情況出現(xiàn)。
為什么要進(jìn)行限流
在高并發(fā)的場景下,如果不進(jìn)行限流,系統(tǒng)可能會因?yàn)檫^多的請求而崩潰。限流可以保護(hù)系統(tǒng)免于被流量打崩,從而保證系統(tǒng)的可用性和穩(wěn)定性。
Gin框架的限流實(shí)現(xiàn)
Gin 是一個基于 Go 語言的 web 框架,它提供了很多方便的中間件,可以方便地實(shí)現(xiàn)限流。
以下是一個基于 Gin 實(shí)現(xiàn)的令牌桶限流的例子:
定義令牌桶結(jié)構(gòu)體
type TokenBucket struct { capacity int64 // 桶的容量 rate float64 // 令牌放入速率 tokens float64 // 當(dāng)前令牌數(shù)量 lastToken time.Time // 上一次放令牌的時間 mtx sync.Mutex // 互斥鎖 }實(shí)現(xiàn)令牌桶算法
func (tb *TokenBucket) Allow() bool { tb.mtx.Lock() defer tb.mtx.Unlock() now := time.Now() // 計(jì)算需要放的令牌數(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 } }使用中間件進(jìn)行限流
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")
以上代碼實(shí)現(xiàn)了一個簡單的令牌桶限流中間件,可以限制最大并發(fā)連接數(shù)為 100。如果超過了這個連接數(shù),將會返回 503 狀態(tài)碼。
測試
瀏覽器地址欄輸入http://localhost:8080/, 然后瘋狂刷新即可.
- 測試截圖

總結(jié)
總的來說,使用 Gin 框架進(jìn)行限流是一個方便有效的方法,可以提高系統(tǒng)的可用性和穩(wěn)定性,避免因?yàn)檫^多的請求導(dǎo)致系統(tǒng)崩潰的問題。利用令牌桶算法實(shí)現(xiàn)限流可以很好地控制請求的并發(fā)量,可以通過控制桶容量和放入速率等參數(shù)進(jìn)行調(diào)節(jié)和優(yōu)化。在使用中間件進(jìn)行限流時,應(yīng)該根據(jù)實(shí)際應(yīng)用場景和需求調(diào)節(jié)限流參數(shù),祝您的應(yīng)用愉快運(yùn)行!
完整代碼
package main
import (
"github.com/gin-gonic/gin"
"sync"
"time"
)
type TokenBucket struct {
capacity int64 // 桶的容量
rate float64 // 令牌放入速率
tokens float64 // 當(dāng)前令牌數(shù)量
lastToken time.Time // 上一次放令牌的時間
mtx sync.Mutex // 互斥鎖
}
func (tb *TokenBucket) Allow() bool {
tb.mtx.Lock()
defer tb.mtx.Unlock()
now := time.Now()
// 計(jì)算需要放的令牌數(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")
}到此這篇關(guān)于Gin框架限流實(shí)現(xiàn)示例的文章就介紹到這了,更多相關(guān)Gin框架限流內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
go語言中基本數(shù)據(jù)類型及應(yīng)用快速了解
這篇文章主要為大家介紹了go語言中基本數(shù)據(jù)類型應(yīng)用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
如何判斷Golang接口是否實(shí)現(xiàn)的操作
這篇文章主要介紹了如何判斷Golang接口是否實(shí)現(xiàn)的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12
golang gin 監(jiān)聽rabbitmq隊(duì)列無限消費(fèi)的案例代碼
這篇文章主要介紹了golang gin 監(jiān)聽rabbitmq隊(duì)列無限消費(fèi),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-12-12
Go結(jié)合MQTT實(shí)現(xiàn)通信的示例代碼
本文主要介紹了Go結(jié)合MQTT實(shí)現(xiàn)通信的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05
golang調(diào)試bug及性能監(jiān)控方式實(shí)踐總結(jié)
這篇文章主要為大家介紹了golang調(diào)試bug及性能監(jiān)控方式實(shí)踐是總結(jié),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05

