go限流、计数器固定窗口算法/计数器滑动窗口算法

这篇具有很好参考价值的文章主要介绍了go限流、计数器固定窗口算法/计数器滑动窗口算法。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

go限流、计数器固定窗口算法/计数器滑动窗口算法

一、问题

问题1:后端接口只能支撑每10秒1w个请求,要怎么来保护它呢?
问题2:发短信的接口,不超过100次/时,1000次/24小时,要怎么实现?

二、计数器固定窗口算法

所谓固定窗口,就是只设置了一个时间段,给这个时间段加上一个计数器。 常见的就是统计每秒钟的请求量。 这里就是一个QPS计数器。 在这一秒种内的所有请求,只要给这个计数器递增就可以得到当前的并发量了。 用这个方法也就可以解决前面的问题1。可以直接使用系统的当前UNIX时间戳,精确到秒钟。 这个时间戳作为key,设置一个较短的过期时间,比如:10s。

package main

import (
	"fmt"
	"time"
)

type SlidingWindow struct {
	WindowSize        int
	Window            []int
	LastUpdateTime    time.Time
	WindowDuration    time.Duration
	CurrentIndex      int
	Count             int
	MaxAllowedRequest int
}

func NewSlidingWindow(windowSize int, windowDuration time.Duration, maxRequest int) *SlidingWindow {
	if windowSize <= 0 {
		panic("windowSize must be greater than 0")
	}
	return &SlidingWindow{
		WindowSize:        windowSize,
		Window:            make([]int, windowSize),
		LastUpdateTime:    time.Now(),
		WindowDuration:    windowDuration,
		CurrentIndex:      0,
		MaxAllowedRequest: maxRequest,
	}
}

func (sw *SlidingWindow) IncrementCount() {
	now := time.Now()

	if now.Sub(sw.LastUpdateTime) >= sw.WindowDuration { //比较时间是否超过限定时间
		sw.ResetWindow()
	}

	sw.Count++
	sw.Window[sw.CurrentIndex] = sw.Count //窗口总量加1

	if sw.Count > sw.MaxAllowedRequest { //最好用窗口总量去计算。这里为了显示两种效果
		fmt.Println("Max allowed request exceeded")
	}
	sw.CurrentIndex = (sw.CurrentIndex + 1) % sw.WindowSize //窗口往后移动一个位置
}

func (sw *SlidingWindow) ResetWindow() {
	for i := range sw.Window {//将窗口归零
		sw.Window[i] = 0
	}
	sw.Count = 0
	sw.LastUpdateTime = time.Now() //记录最后一次更新
}

func main2() {
	windowSize := 5 //窗口粒度,问题1的话可以简化掉。
	windowDuration := time.Second * 10//十秒的访问量
	maxRequest := 10000//最大访问量

	slidingWindow := NewSlidingWindow(windowSize, windowDuration, maxRequest)

	for i := 0; i < 10; i++ {
		slidingWindow.IncrementCount()
		time.Sleep(time.Second)
	}
}

三、计数器滑动窗口算法

固定窗口就一个计数器,而滑动窗口就需要有多个计数器。 具体需要多少个计数器,要看窗口的范围和粒度来决定窗口大小。 比如:时间窗口的范围是24小时,时间窗口的粒度是1小时,那么窗口大小就是24,需要的计数器也就是24个。 我们再来回顾下前面的问题2。 如果我们用上面的固定窗口算法,需要2个计数器,一个是小时的计数器,一个是24小时,也就是天的计数器。 很明显,天的计数器会有很大的误差。 比如:昨天14点前没有任何请求,然后在14点开始,每小时都有100次请求。 到昨天的23点,刚好用完了1000次全天的额度。 但是这时候,还是每小时有100个请求, 那么从昨天的14点到今天10点,总共20小时就会有2000次请求,远远超过了24小时最多1500次的限制。 所以,这里使用滑动窗口替代固定窗口会更加合适。 如果想要限流控制点更加精准,那么就可以把窗口粒度设计的更细。 而代价就是窗口大小增加,需要的存储和计算量都会增加。 所以,这里也是需要对精准度和成本做平衡和选择,难以兼得。
qps 控制 滑动窗口 go,golang,算法文章来源地址https://www.toymoban.com/news/detail-858218.html

package main

import (
	"fmt"
	"time"
)

type RateLimiter struct {
	perHour     int
	perDay      int
	hourWindow  []int
	dayWindow   []int
	lastHourIdx int
	lastDayIdx  int
}

func NewRateLimiter(perHour, perDay int) *RateLimiter {
	return &RateLimiter{
		perHour:     perHour,
		perDay:      perDay,
		hourWindow:  make([]int, 60),
		dayWindow:   make([]int, 1440),
		lastHourIdx: 0,
		lastDayIdx:  0,
	}
}

func (rl *RateLimiter) Allow() bool {
	now := time.Now()
	hourIdx := (now.Minute() + now.Hour()*60) % 60  //记录小时的id,
	dayIdx := now.Hour()*60 + now.Minute()         //记录天的id

	if rl.hourWindow[hourIdx] >= rl.perHour || rl.dayWindow[dayIdx] >= rl.perDay {
		return false
	}

	rl.hourWindow[hourIdx]++
	rl.dayWindow[dayIdx]++
	rl.cleanUpOldEntries(hourIdx, dayIdx)

	return true
}

func (rl *RateLimiter) cleanUpOldEntries(hourIdx, dayIdx int) {
	if hourIdx != rl.lastHourIdx {//如果小时id更新了需要窗口往右移动
		rl.hourWindow[hourIdx] = 1 //最新小时id的总量为1
		rl.hourWindow[rl.lastHourIdx] = 0 //窗口往右移动,上一个归零
		rl.lastHourIdx = hourIdx //记录最新id
	}

	if dayIdx != rl.lastDayIdx {
		rl.dayWindow[dayIdx] = 1
		rl.dayWindow[rl.lastDayIdx] = 0
		rl.lastDayIdx = dayIdx
	}
}

func main3() {
	limiter := NewRateLimiter(100, 1000)

	for i := 0; i < 1200; i++ {
		if limiter.Allow() {
			fmt.Printf("Request %d allowed\n", i+1)
		} else {
			fmt.Printf("Request %d blocked\n", i+1)
		}
		time.Sleep(time.Second)
	}
}

到了这里,关于go限流、计数器固定窗口算法/计数器滑动窗口算法的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • JAVA:基于Redis 实现计数器限流

    JAVA:基于Redis 实现计数器限流

    1、简述 在现实世界中可能会出现服务器被虚假请求轰炸的情况,因此您可能希望控制这种虚假的请求。 一些实际使用情形可能如下所示: API配额管理-作为提供者,您可能希望根据用户的付款情况限制向服务器发出API请求的速率。这可以在客户端或服务端实现。 安全性-防止

    2024年02月02日
    浏览(10)
  • springboot 自定义注解 ,实现接口限流(计数器限流)【强行喂饭版】

    springboot 自定义注解 ,实现接口限流(计数器限流)【强行喂饭版】

    思路:通过AOP拦截注解标记的方法,在Redis中维护一个计数器来记录接口访问的频率, 并根据限流策略来判断是否允许继续处理请求。 另一篇:springboot 自定义注解 ,aop切面@Around; 为接口实现日志插入【强行喂饭版】 不多说,直接上代码: 一:创建限流类型 二:创建注解

    2024年02月15日
    浏览(13)
  • 【FPGA】Verilog:计数器 | 异步计数器 | 同步计数器 | 2位二进制计数器的实现 | 4位十进制计数器的实现

    【FPGA】Verilog:计数器 | 异步计数器 | 同步计数器 | 2位二进制计数器的实现 | 4位十进制计数器的实现

    目录 Ⅰ. 实践说明 0x00 计数器(Counter) 0x01 异步计数器(Asynchronous Counter)

    2024年02月05日
    浏览(15)
  • 【FPGA】Verilog:升降计数器 | 波纹计数器 | 约翰逊计数器 | 实现 4-bit 升降计数器的 UP/DOWN

    【FPGA】Verilog:升降计数器 | 波纹计数器 | 约翰逊计数器 | 实现 4-bit 升降计数器的 UP/DOWN

    目录 Ⅰ. 理论部分 0x00 升降计数器(UP DOWN Counter) 0x01 波纹计数器(Ripple Counter)

    2024年02月05日
    浏览(14)
  • FPGA拾忆_(3):调用IP 计数器&BCD计数器

    FPGA拾忆_(3):调用IP 计数器&BCD计数器

    调用IP计数器: 每来一个cin(进位输入)信号,计数器输出值加一,当计数值为9且cin为1时,输出一个时钟长度的cout(进位输出)信号。 首先采用调用quartus种IP的方式,具体步骤: Tools----IP Catalog: 然后会调出IP目录窗口: 通过搜索counter来添加计数器模块,需要设置的内容

    2024年02月03日
    浏览(9)
  • verilog手撕代码5——计数器(置位、加减、环形、扭环形、格雷码计数器实现)

    verilog手撕代码5——计数器(置位、加减、环形、扭环形、格雷码计数器实现)

    2023.5.12 编写一个十六进制计数器模块,计数器输出信号递增每次到达0,给出指示信号 zero ,当置位信号 set 有效时,将当前输出置为输入的数值 set_num 。 注意 :这里zero=1和num=0是同一拍输出的,按道理如果根据num=0,然后去输出zero=1应该延迟一拍。所以这里考虑将number延迟一

    2024年02月07日
    浏览(11)
  • LR中监控ORACLE数据库常用计数器(如何自定义Oracle计数器)

    目录 一、添加自定义计数器的方法 1、要创建自定义查询,请执行以下操作: 2、配置文件示例对象 二、常用自定义计数器列表 三、LR中监控ORACLE数据库常用计数器遇到问题及处理 1. 在安装路径的Mercury LoadRunnerdatmonitors找到vmon.cfg文件,打开。 2. 在vmon.cfg文件的第三行中,

    2024年02月15日
    浏览(11)
  • 【FPGA】Verilog:时序电路设计 | 二进制计数器 | 计数器 | 分频器 | 时序约束

    【FPGA】Verilog:时序电路设计 | 二进制计数器 | 计数器 | 分频器 | 时序约束

    前言: 本章内容主要是演示Vivado下利用Verilog语言进行电路设计、仿真、综合和下载 示例:计数器与分频器   ​​ 功能特性: 采用 Xilinx Artix-7 XC7A35T芯片  配置方式:USB-JTAG/SPI Flash 高达100MHz 的内部时钟速度  存储器:2Mbit SRAM   N25Q064A SPI Flash(样图旧款为N25Q032A) 通用

    2024年02月02日
    浏览(15)
  • 用74LS73设计四位二进制加法计数器和8421BCD加法计数器

    用74LS73设计四位二进制加法计数器和8421BCD加法计数器

     (1)用2片74LS73实现该电路,由CP端输入单脉冲,设计并画出4位异步二进制加法计数器电路图。  (2)由CP端输入单脉冲,测试并记录Q1~Q4端状态及波形。 四位二进制加法计数器状态迁移表如下: Q 4n Q 3n Q 2n Q 1n Q 4n+1 Q 3n+1 Q 2n+1 Q 1n+1 0 0 0 0 0 0 0 1 0 0 0 1 0 0 1 0 0 0 1 0 0 0 1 1 0 0 1 1 0 1 0

    2024年02月10日
    浏览(39)
  • 定时器/计数器中定时/计数初值的计算

    定时器/计数器中定时/计数初值的计算

             寄存器TMOD是单片机的一个特殊功能寄存器,其功能是控制定时器/计数器T0、T1的工作方式。它的字节地址为89H, 不可以对它进行位操作。          只能进行字节操作, 即给寄存器整体赋值的方法 设置初始值 ,如 TMOD=0x01 。在上电和复位时,寄存器TMOD的初始

    2024年02月10日
    浏览(9)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包