Gin框架限流实现示例

 更新时间:2023年03月19日 11:11:51   作者:梧桐知秋  
本文主要介绍了Gin框架限流实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

什么是限流

限流是指通过一定的算法,对接口的请求进行限制,防止并发量过大,导致系统瘫痪或响应变慢的情况出现。

为什么要进行限流

在高并发的场景下,如果不进行限流,系统可能会因为过多的请求而崩溃。限流可以保护系统免于被流量打崩,从而保证系统的可用性和稳定性。

Gin框架的限流实现

Gin 是一个基于 Go 语言的 web 框架,它提供了很多方便的中间件,可以方便地实现限流。

以下是一个基于 Gin 实现的令牌桶限流的例子:

  • 定义令牌桶结构体

    type TokenBucket struct {
        capacity  int64   // 桶的容量
        rate      float64 // 令牌放入速率
        tokens    float64 // 当前令牌数量
        lastToken time.Time // 上一次放令牌的时间
        mtx       sync.Mutex // 互斥锁
    }
  • 实现令牌桶算法

    func (tb *TokenBucket) Allow() bool {
        tb.mtx.Lock()
        defer tb.mtx.Unlock()
        now := time.Now()
        // 计算需要放的令牌数量
        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")

以上代码实现了一个简单的令牌桶限流中间件,可以限制最大并发连接数为 100。如果超过了这个连接数,将会返回 503 状态码。

测试

浏览器地址栏输入http://localhost:8080/, 然后疯狂刷新即可.

  • 测试截图

总结

总的来说,使用 Gin 框架进行限流是一个方便有效的方法,可以提高系统的可用性和稳定性,避免因为过多的请求导致系统崩溃的问题。利用令牌桶算法实现限流可以很好地控制请求的并发量,可以通过控制桶容量和放入速率等参数进行调节和优化。在使用中间件进行限流时,应该根据实际应用场景和需求调节限流参数,祝您的应用愉快运行!

完整代码

package main

import (
   "github.com/gin-gonic/gin"
   "sync"
   "time"
)

type TokenBucket struct {
   capacity  int64      // 桶的容量
   rate      float64    // 令牌放入速率
   tokens    float64    // 当前令牌数量
   lastToken time.Time  // 上一次放令牌的时间
   mtx       sync.Mutex // 互斥锁
}

func (tb *TokenBucket) Allow() bool {
   tb.mtx.Lock()
   defer tb.mtx.Unlock()
   now := time.Now()
   // 计算需要放的令牌数量
   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框架限流实现示例的文章就介绍到这了,更多相关Gin框架限流内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Golang的select多路复用及channel使用操作

    Golang的select多路复用及channel使用操作

    这篇文章主要介绍了Golang的select多路复用及channel使用操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • Windows下Goland的环境搭建过程详解

    Windows下Goland的环境搭建过程详解

    这篇文章主要介绍了Windows下Goland的环境搭建过程,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-10-10
  • 详解在Go语言单元测试中如何解决Redis存储依赖问题

    详解在Go语言单元测试中如何解决Redis存储依赖问题

    在编写单元测试时,除了 MySQL 这个外部存储依赖,Redis 应该是另一个最为常见的外部存储依赖了,本文就来讲解下如何解决 Redis 外部依赖,文章通过代码示例介绍的非常详细,需要的朋友可以参考下
    2023-08-08
  • golang 函数以及函数和方法的详解及区别

    golang 函数以及函数和方法的详解及区别

    这篇文章主要介绍了golang 函数以及函数和方法的区别的相关资料,需要的朋友可以参考下
    2017-05-05
  • 实用的Go语言开发工具及使用示例

    实用的Go语言开发工具及使用示例

    这篇文章主要为大家介绍了实用的Go语言开发工具及使用示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • Go语言下载网络图片或文件的方法示例

    Go语言下载网络图片或文件的方法示例

    这篇文章主要介绍了Go语言下载网络图片或文件的方法示例,文中通过示例代码介绍的非常详细,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • 解决Golang中goroutine执行速度的问题

    解决Golang中goroutine执行速度的问题

    这篇文章主要介绍了解决Golang中goroutine执行速度的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-05-05
  • golang 一次性定时器Timer用法及实现原理详解

    golang 一次性定时器Timer用法及实现原理详解

    这篇文章主要为大家介绍了golang 一次性定时器Timer用法及实现原理详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • 基于微服务框架go-micro开发gRPC应用程序

    基于微服务框架go-micro开发gRPC应用程序

    这篇文章介绍了基于微服务框架go-micro开发gRPC应用程序的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-07-07
  • grpc-go如何通过context传递额外数据

    grpc-go如何通过context传递额外数据

    metadata是grpc内置的,用RPC服务传递http头数据,分in和out两种,对应的key都为一个空struct,这篇文章主要介绍了grpc-go通过context传递额外数据,需要的朋友可以参考下
    2024-02-02

最新评论