av一区二区在线观看_亚洲男人的天堂网站_日韩亚洲视频_在线成人免费_欧美日韩精品免费观看视频_久草视

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

springboot+redis 實(shí)現(xiàn)分布式限流令牌桶的示例代碼

瀏覽:133日期:2023-03-14 15:36:59
1、前言

網(wǎng)上找了很多redis分布式限流方案,要不就是太大,需要引入第三方j(luò)ar,而且還無(wú)法正常運(yùn)行,要不就是定時(shí)任務(wù)定時(shí)往key中放入數(shù)據(jù),使用的時(shí)候調(diào)用,嚴(yán)重影響性能,所以著手自定義實(shí)現(xiàn)redis令牌桶。只用到了spring-boot-starter-data-redis包,并且就幾行代碼。

2、環(huán)境準(zhǔn)備

a、idea新建springboot項(xiàng)目,引入spring-data-redis包b、編寫(xiě)令牌桶實(shí)現(xiàn)方法RedisLimitExcutorc、測(cè)試功能,創(chuàng)建全局?jǐn)r截器,測(cè)試功能

3、上代碼

springboot+redis 實(shí)現(xiàn)分布式限流令牌桶的示例代碼

maven添加依賴

<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId></dependency>

令牌桶實(shí)現(xiàn)方法RedisLimitExcutor

package com.example.redis_limit_demo.config;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.data.redis.core.StringRedisTemplate;import org.springframework.data.redis.core.script.DefaultRedisScript;import org.springframework.data.redis.core.script.RedisScript;import org.springframework.stereotype.Component;import java.util.ArrayList;import java.util.List;/** * 令牌桶實(shí)現(xiàn) */@Componentpublic class RedisLimitExcutor { private StringRedisTemplate stringRedisTemplate; @Autowired public void setStringRedisTemplate(StringRedisTemplate stringRedisTemplate) {this.stringRedisTemplate = stringRedisTemplate; } /** * 令牌的 * * @param keykey值 * @param limitCount 容量 * @param seconds 時(shí)間間隔 * @return */ public boolean tryAccess(String key, int limitCount, int seconds) {String luaScript = buildLuaScript();RedisScript<Long> redisScript = new DefaultRedisScript<>(luaScript, Long.class);List<String> keys = new ArrayList<>();keys.add(key);Long count = stringRedisTemplate.execute(redisScript, keys, String.valueOf(limitCount), String.valueOf(seconds));if (count != 0) { return true;} else { return false;} } /** * 腳本 * * @return */ private static final String buildLuaScript() {StringBuilder lua = new StringBuilder();lua.append(' local key = KEYS[1]');lua.append('nlocal limit = tonumber(ARGV[1])');lua.append('nlocal curentLimit = tonumber(redis.call(’get’, key) or '0')');lua.append('nif curentLimit + 1 > limit then');lua.append('nreturn 0');lua.append('nelse');lua.append('n redis.call('INCRBY', key, 1)');lua.append('nredis.call('EXPIRE', key, ARGV[2])');lua.append('nreturn curentLimit + 1');lua.append('nend');return lua.toString(); }}

攔截器配置WebAppConfig

package com.example.redis_limit_demo.config;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.web.servlet.config.annotation.InterceptorRegistry;import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;/** * 攔截器配置 */@Configurationpublic class WebAppConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(getRequestInterceptor()).addPathPatterns('/**'); } @Bean public RequestInterceptor getRequestInterceptor() {return new RequestInterceptor(); }}

攔截器實(shí)現(xiàn)RequestInterceptor

package com.example.redis_limit_demo.config;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.context.annotation.Configuration;import org.springframework.web.servlet.HandlerInterceptor;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import java.net.InetAddress;import java.net.UnknownHostException;/** * 攔截器實(shí)現(xiàn) */@Configurationpublic class RequestInterceptor implements HandlerInterceptor { @Autowired private RedisLimitExcutor redisLimitExcutor; /** * 只有返回true才會(huì)繼續(xù)向下執(zhí)行,返回false取消當(dāng)前請(qǐng)求 * * @param request * @param response * @param handler * @return */ @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {/** * 根據(jù)實(shí)際情況設(shè)置QPS */String url = request.getRequestURI();String ip = getIpAdd(request);//QPS設(shè)置為5,手動(dòng)刷新接口可以測(cè)試出來(lái)if (!redisLimitExcutor.tryAccess(ip+url, 5, 1)) { throw new RuntimeException('調(diào)用頻繁');} else { return true;} } public static final String getIpAdd(HttpServletRequest request) {String ipAddress = request.getHeader('x-forwarded-for');if (ipAddress == null || ipAddress.length() == 0 || 'unknown'.equalsIgnoreCase(ipAddress)) { ipAddress = request.getHeader('Proxy-Client-IP');}if (ipAddress == null || ipAddress.length() == 0 || 'unknown'.equalsIgnoreCase(ipAddress)) { ipAddress = request.getHeader('WL-Proxy-Client-IP');}if (ipAddress == null || ipAddress.length() == 0 || 'unknown'.equalsIgnoreCase(ipAddress)) { ipAddress = request.getRemoteAddr(); if (ipAddress.equals('127.0.0.1') || ipAddress.equals('0:0:0:0:0:0:0:1')) {// 根據(jù)網(wǎng)卡取本機(jī)配置的IPInetAddress inet = null;try { inet = InetAddress.getLocalHost();} catch (UnknownHostException e) { return null;}ipAddress = inet.getHostAddress(); }}// 如果通過(guò)代理訪問(wèn),可能獲取2個(gè)IP,這時(shí)候去第二個(gè)(代理服務(wù)端IP)if (ipAddress.split(',').length > 1) { ipAddress = ipAddress.split(',')[1].trim();}return ipAddress; }}

測(cè)試controller

package com.example.redis_limit_demo.controller;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;@RequestMapping('demo')@RestControllerpublic class DemoController { @RequestMapping('limit') public String demo() {//todo 寫(xiě)業(yè)務(wù)邏輯return 'aaaaa'; }}4、運(yùn)行項(xiàng)目,訪問(wèn)接口

http://localhost:8080/demo/limit

springboot+redis 實(shí)現(xiàn)分布式限流令牌桶的示例代碼

當(dāng)刷新頻率高了以后,就會(huì)報(bào)錯(cuò)

5、碼云地址(GitHub經(jīng)常訪問(wèn)不到)

備注:

1、 redis的key可以根據(jù)實(shí)際情況設(shè)置,入例子中的ip+url,可以將全部流量進(jìn)行控制,防止惡意刷接口,但需要注意的是,使用ip方式,要將QPS設(shè)置大一些,因?yàn)闀?huì)出現(xiàn)整個(gè)大廈公用一個(gè)ip的情況。也可以使用url+userName,將QPS設(shè)置小一點(diǎn),可以更加精準(zhǔn)的限制api的訪問(wèn)。2、可以將拋出異常進(jìn)行全局捕獲和統(tǒng)一返回。

到此這篇關(guān)于springboot+redis 實(shí)現(xiàn)分布式限流令牌桶的示例代碼的文章就介紹到這了,更多相關(guān)springboot redis分布式限流令牌桶內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Spring
相關(guān)文章:
主站蜘蛛池模板: 韩国毛片视频 | 亚洲成网站 | 久久精品国产99国产精品 | 一本色道久久综合亚洲精品高清 | 大学生a级毛片免费视频 | 午夜视频在线播放 | 久久精品国产99国产精品 | 欧美视频二区 | 天天操天天操 | 亚洲视频在线观看 | 日韩成人在线视频 | 亚洲成人第一页 | 一区二区国产精品 | 99精品视频免费观看 | 国产网站在线免费观看 | 亚洲二区在线 | 涩爱av一区二区三区 | 国产精品久久久久久久久免费樱桃 | 欧美精品一二三区 | 热久久久久 | 国产传媒在线播放 | 精品国产伦一区二区三区观看方式 | 日韩毛片 | 婷婷久久久久 | 久久青青 | 亚洲欧美视频 | 日本黄色一级片视频 | 日韩在线精品视频 | 亚洲国产aⅴ精品一区二区 免费观看av | 亚洲一在线 | 亚洲激情在线视频 | 一区二区精品在线 | 蜜桃日韩| 人人鲁人人莫人人爱精品 | 欧美一级久久 | 四季久久免费一区二区三区四区 | 亚洲成人精品 | 亚洲精品一区二区在线观看 | www国产成人免费观看视频,深夜成人网 | 午夜羞羞| 成人精品一区亚洲午夜久久久 |