Go中的动态速率限制有效控制流量

 更新时间:2023年10月17日 11:20:58   作者:TimLiu  
这篇文章主要为大家介绍了Go中的动态速率限制有效控制流量,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

介绍

速率限制是构建可扩展和弹性系统的关键技术。它通过对指定时间范围内允许的请求数量施加限制来帮助控制流量。在Go中实现速率限制可以确保最佳的资源利用,并保护您的应用程序免受过度流量或滥用行为的影响。在这篇博文中,我们将探索Go中的速率限制技术,并提供实用的代码示例来帮助您有效地实现它们。

理解速率限制

速率限制包括定义一组规则,这些规则决定客户端在给定的时间窗口内可以发出多少请求。确保系统能够处理负载,防止滥用或拒绝服务攻击。限制速率的两种常见方法是:

  • 固定窗口速率限制:在这种方法中,速率限制在固定的时间窗口内强制执行。例如,如果速率限制设置为每分钟100个请求,系统将在任何给定的60秒窗口中允许最多100个请求。超过此限制的请求将被拒绝或延迟到下一个时间窗口。
  • 令牌桶速率限制:令牌桶速率限制是基于从一个桶中消耗令牌的概念。桶最初由固定数量的令牌填充,每个令牌代表一个请求。当客户端想要发出请求时,它必须从桶中获取令牌。如果桶为空,客户端必须等待,直到令牌可用。

在GO中实现速率限制

Go提供了一个名为golang.org/x/time/rate的内置包,提供速率限制功能。让我们探讨一下如何同时使用固定窗口和令牌桶方法来实现速率限制。

固定窗口

func fixedWindowRateLimiting() {
    limiter := rate.NewLimiter(rate.Limit(100), 1) // 允许每秒100次
    for i := 0; i < 200; i++ {
        if !limiter.Allow() {
            fmt.Println("Rate limit exceeded. Request rejected.")
            continue
        }
        go process()
    }
}
// 处理请求
func process() {
    fmt.Println("Request processed successfully.")
    time.Sleep(time.Millisecond) // 模拟请求处理时间
}

在上面的代码片段中,我们创建了一个限制使用率。速率限制为每秒100个请求的NewLimiter。对每个请求调用limiter.Allow()方法,如果允许请求,则返回true;如果超过速率限制,则返回false。如果超过速率限制,请求将被拒绝。

对应的输出为,清楚的看到部分请求已经被拒绝了:

Request processed successfully.
Rate limit exceeded. Request rejected.
Rate limit exceeded. Request rejected.
Rate limit exceeded. Request rejected.
Rate limit exceeded. Request rejected.
Rate limit exceeded. Request rejected.
Rate limit exceeded. Request rejected.
Rate limit exceeded. Request rejected.
...

令牌桶

func tokenBucketRateLimiting() {
    limiter := rate.NewLimiter(rate.Limit(10), 5)
    ctx, _ := context.WithTimeout(context.TODO(), time.Millisecond)
    for i := 0; i < 200; i++ {
        if err := limiter.Wait(ctx); err != nil {
            fmt.Println("Rate limit exceeded. Request rejected.")
            continue
        }
        go process()
    }
}

// 处理请求
func process() {
    fmt.Println("Request processed successfully.")
    time.Sleep(time.Millisecond) // 模拟请求处理时间
}

在上述代码中,我们使用 rate.NewLimiter 创建了一个限制器,其速率限制为每秒 10 个请求,突发 5 个请求。每个请求都会调用 limiter.Wait() 方法,该方法会阻塞直到有令牌可用。如果水桶是空的,没有可用的令牌,请求就会被拒绝。

对应的输出为,清楚的看到部分请求已经被拒绝了:

Request processed successfully.
Rate limit exceeded. Request rejected.
Rate limit exceeded. Request rejected.
Request processed successfully.
Rate limit exceeded. Request rejected.

动态速率限制

动态速率限制是指根据客户端行为、系统负载或业务规则等动态因素调整速率限制。这种技术允许您实时调整速率限制,以优化资源利用率并提供更好的用户体验。让我们看看 Go 中动态速率限制的示例:

func dynamicRateLimiting() {
    limiter := rate.NewLimiter(rate.Limit(10), 1) // 允许每秒100次
    // Dynamic rate adjustment
    go func() {
        time.Sleep(time.Second * 10) // 每10秒调整 limiter
        fmt.Println("---adjust limiter---")
        limiter.SetLimit(rate.Limit(200)) // 将 limiter 提升到每秒 200
    }()
    for i := 0; i < 3000; i++ {
        if !limiter.Allow() {
            fmt.Println("Rate limit exceeded. Request rejected.")
            time.Sleep(time.Millisecond * 100)
            continue
        }
        process()
    }
}
// 处理请求
func process() {
    fmt.Println("Request processed successfully.")
    time.Sleep(time.Millisecond * 10) // 模拟请求处理时间
}

在上面的代码片段中,我们创建了一个限制器,初始速率限制为每秒 10 个请求。然后,我们启动一个 goroutine,在 10秒钟后将速率限制调整为每秒 200 个请求。这样,我们就能根据不断变化的情况动态调整速率限制。

对应的输出为,这里我们可以看到中途的时候,某些请求已经被拒绝掉了,后来我们通过动态调整,后续的请求都可以正常通过了:

...
Request processed successfully.
Rate limit exceeded. Request rejected.
Request processed successfully.
---adjust limiter---
Rate limit exceeded. Request rejected.
Request processed successfully.
Request processed successfully.
Request processed successfully.
Request processed successfully.
Request processed successfully.
...

自适应速率限制

自适应速率限制可根据之前请求的响应时间或错误率动态调整速率限制。它允许系统自动适应不同的流量条件,确保最佳性能和资源利用率。让我们看看 Go 中自适应速率限制的示例:

func adaptiveRateLimiting() {
    limiter := rate.NewLimiter(rate.Limit(10), 1) // 允许每秒10次
    // 自适应调整
    go func() {
        for {
            time.Sleep(time.Second * 10)
            responseTime := measureResponseTime() // 测量之前请求的响应时间
            if responseTime > 500*time.Millisecond {
                fmt.Println("---adjust limiter 50---")
                limiter.SetLimit(rate.Limit(50))
            } else {
                fmt.Println("---adjust limiter 100---")
                limiter.SetLimit(rate.Limit(100))
            }
        }
    }()
    for i := 0; i < 3000; i++ {
        if !limiter.Allow() {
            fmt.Println("Rate limit exceeded. Request rejected.")
            time.Sleep(time.Millisecond * 100)
            continue
        }
        process()
    }
}
// 测量以前请求的响应时间
// 执行自己的逻辑来测量响应时间
func measureResponseTime() time.Duration {
    return time.Millisecond * 100
}
// 处理请求
func process() {
    fmt.Println("Request processed successfully.")
    time.Sleep(time.Millisecond * 10) // 模拟请求处理时间
}

在上述代码片段中,我们使用 measureResponseTime 函数模拟测量之前请求的响应时间。根据测量到的响应时间,我们通过使用 limiter.SetLimit 设置不同的值来动态调整速率限制。这样,系统就能根据观察到的响应时间调整其速率限制策略。

对应的输出为:

Request processed successfully.
Rate limit exceeded. Request rejected.
Request processed successfully.
Rate limit exceeded. Request rejected.
---adjust limiter 100---
Request processed successfully.
Request processed successfully.
Request processed successfully.
Request processed successfully.
Request processed successfully.

总结

速率限制是维护 Go 应用程序稳定性和安全性的基本技术。通过有效控制传入请求的流量,您可以防止资源耗尽并确保资源的公平分配。在这篇博文中,我们探讨了固定窗口和令牌桶速率限制的概念,并提供了代码片段,演示如何使用 golang.org/x/time/rate 包在 Go 中实现它们。将速率限制纳入您的应用程序,以构建能够高效处理不同流量水平的弹性系统。

当然可以使用第三方库来实现,比如说:go.uber.org/ratelimit

以上就是Go中的动态速率限制有效控制流量的详细内容,更多关于Go流量动态控制的资料请关注脚本之家其它相关文章!

相关文章

  • Go实现set类型的示例代码

    Go实现set类型的示例代码

    本文主要介绍了Go实现set类型的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-01-01
  • Go语言状态机的实现

    Go语言状态机的实现

    本文主要介绍了Go语言状态机的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-03-03
  • 使用Go语言写一个Http Server的实现

    使用Go语言写一个Http Server的实现

    本文主要介绍了使用Go语言写一个Http Server的实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-04-04
  • Go函数使用(函数定义、函数声明、函数调用等)

    Go函数使用(函数定义、函数声明、函数调用等)

    本文主要介绍了Go函数使用,包括函数定义、函数声明、函数调用、可变参数函数、匿名函数、递归函数、高阶函数等,感兴趣的可以了解一下
    2023-11-11
  • Go之集合slice的实现

    Go之集合slice的实现

    本文主要介绍了Go之集合slice的实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • Go使用Weighted实现资源管理

    Go使用Weighted实现资源管理

    这篇文章主要介绍了Go 语言中的 Weighted 并发原语,包括 Weighted 的基本使用方法、实现原理、使用注意事项等内容,感兴趣的小伙伴可以了解一下
    2023-06-06
  • Go语言中定时任务库Cron使用方法介绍

    Go语言中定时任务库Cron使用方法介绍

    cron的意思计划任务,说白了就是定时任务。我和系统约个时间,你在几点几分几秒或者每隔几分钟跑一个任务(job),今天通过本文给大家介绍下Go语言中定时任务库Cron使用方法,感兴趣的朋友一起看看吧
    2022-03-03
  • 小学生也能看懂的Golang异常处理recover panic

    小学生也能看懂的Golang异常处理recover panic

    在其他语言里,宕机往往以异常的形式存在,底层抛出异常,上层逻辑通过 try/catch 机制捕获异常,没有被捕获的严重异常会导致宕机,go语言追求简洁,优雅,Go语言不支持传统的 try…catch…finally 这种异常
    2021-09-09
  • Golang Configor配置文件工具的使用详解

    Golang Configor配置文件工具的使用详解

    Configor是一个支持 yaml、json、toml、shell 的配置文件工具,这篇文中主要为大家详细介绍了Configor的具体使用,感兴趣的小伙伴可以学习一下
    2023-08-08
  • Go 代码块作用域变量遮蔽问题解析

    Go 代码块作用域变量遮蔽问题解析

    这篇文章主要为大家介绍了Go 代码块作用域变量遮蔽问题解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-10-10

最新评论