RedisTemplate实现令牌桶限流

令牌桶

Redisson可以实现很多东西,在Redis的基础上,Redisson做了超多的封装,不仅可以用来实现分布式锁,还可以帮助我们实现令牌桶限流。

Ratelimter主要作用就是可以限制调用接口的次数。主要原理就是调用接口之前,需要拥有指定个令牌,限流器每秒会产生X个令牌放入令牌桶,调用接口需要去令牌桶里面拿令牌。如果令牌被其它请求拿完了,那么自然而然,当前请求就调用不到指定的接口。

RateLimter实现限流

java 复制代码
@RestController
    @RequestMapping("/redisTest")
    public class RedisTestController {
 
        @Autowired
        private Redisson redisson;
 
 
        @GetMapping("/Token")
        public String testTokenBucket() {
            RRateLimiter rateLimiter = redisson.getRateLimiter("myRatelimiter");
 
            //最大流速 =每10秒钟产生1个令牌
            rateLimiter.trySetRate(RateType.OVERALL, 1, 10, RateIntervalUnit.SECONDS);
 
            //需要1个令牌
            if (rateLimiter.tryAcquire(1)) {
                return "令牌桶里面有可使用的令牌";
            }
            return "不好意思,请过十秒钟再来~~~~~~~";
        }
    }

RedisTemplate实现方式:

话不多说直接上代码,如下:

java 复制代码
package com.shop.cyshop.commons.service;

/**
 * 限流器
 *
 * @date 2024年04月28日 15:02
 */

import com.shop.cyshop.commons.locker.LockService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;

import java.util.concurrent.TimeUnit;

@Service
@Slf4j
public class RequestLimitService {

    private static final String REQUEST_LIMIT_KEY = "request_limit";
    private static final int WINDOW_SIZE_SECONDS = 6; //每六秒一次请求限制
    private static final int MAX_REQUESTS = 1;

    @Autowired
    private RedisTemplate<String, String> redisTemplate;

    @Autowired
    private LockService lockService;

    public boolean isAllowed() {
        return lockService.lock(REQUEST_LIMIT_KEY + "::glock", "sys", () -> {
            long currentTime = System.currentTimeMillis() / 1000; //10
            long windowStart = currentTime - WINDOW_SIZE_SECONDS;//4

            // 删除过期的记录
            redisTemplate.opsForZSet().removeRangeByScore(REQUEST_LIMIT_KEY, 0, windowStart);

        
            // 获取当前窗口内的请求数量
            Long currentRequests = redisTemplate.opsForZSet().count(REQUEST_LIMIT_KEY, windowStart, currentTime);

            if (currentRequests != null && currentRequests < MAX_REQUESTS) {
                // 增加请求记录
                redisTemplate.opsForZSet().add(REQUEST_LIMIT_KEY, String.valueOf(currentTime), currentTime);
                // 设置过期时间
                redisTemplate.expire(REQUEST_LIMIT_KEY, WINDOW_SIZE_SECONDS, TimeUnit.SECONDS);
                return true;
            }
            return false;
        });
    }
}
相关推荐
爬山算法几秒前
Netty(14)如何处理Netty中的异常和错误?
java·前端·数据库
李慕婉学姐18 分钟前
【开题答辩过程】以《基于Android的健康助手APP的设计与实现》为例,不知道这个选题怎么做的,不知道这个选题怎么开题答辩的可以进来看看
android·java·mysql
qq_124987075332 分钟前
基于springboot健康养老APP的设计与实现(源码+论文+部署+安装)
java·spring boot·后端·mysql·微信小程序·毕业设计
张np35 分钟前
java基础-Deque 接口
java·开发语言
骚戴37 分钟前
大语言模型(LLM)进阶:从闭源大模型 API 到开源大模型本地部署,四种接入路径全解析
java·人工智能·python·语言模型·自然语言处理·llm·开源大模型
东华万里39 分钟前
Release 版本禁用 assert:NDEBUG 的底层逻辑与效率优化
java·jvm·算法
silence25044 分钟前
基于 (java) validation-api、hibernate-validator 的数据校验扩展
java
Alsn8644 分钟前
24.idea专业版安装+maven、tomcat安装并部署到idea
java·ide·intellij-idea
胡闹541 小时前
海康和大华厂商的RTSP取流地址格式进行拉流直播
java·网络