可能要用心学高并发核心编程,限流原理与实战,分布式令牌桶限流

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介:   实战:分布式令牌桶限流  本节介绍的分布式令牌桶限流通过Lua+Java结合完成,首先在Lua脚本中完成限流的计算,然后在Java代码中进行组织和调用。  分布式令牌桶限流Lua脚本  分布式令牌桶限流Lua脚本的核心逻辑和Java令牌桶的执行逻辑类似,只是限流计算相关的统计和时间数据存放于Redis中。  这里将限流的脚本命名为rate_limiter.lua,该脚本既使用Redis存储令牌桶信息,自身又执行于Redis中,所以笔者将该脚本放置于base-redis基础模块中,它的代码如下:  ---此脚本的环境:redis内部,不是运行在Nginx内部

  实战:分布式令牌桶限流

  本节介绍的分布式令牌桶限流通过Lua+Java结合完成,首先在Lua脚本中完成限流的计算,然后在Java代码中进行组织和调用。

  分布式令牌桶限流Lua脚本

  分布式令牌桶限流Lua脚本的核心逻辑和Java令牌桶的执行逻辑类似,只是限流计算相关的统计和时间数据存放于Redis中。

  这里将限流的脚本命名为rate_limiter.lua,该脚本既使用Redis存储令牌桶信息,自身又执行于Redis中,所以笔者将该脚本放置于base-redis基础模块中,它的代码如下:

  ---此脚本的环境:redis内部,不是运行在Nginx内部

  ---方法:申请令牌

  ----1:failed

  ---1:success

  ---@param key:key限流关键字

  ---@param apply:申请的令牌数量

  local function acquire(key, apply)

  local times=redis.call('TIME');

  --times[1] 秒数 --times[2] 微秒数

  local curr_mill_second=times[1] *1000000 + times[2];

  curr_mill_second=curr_mill_second / 1000;

  local cacheInfo=redis.pcall("HMGET", key, "last_mill_second", "curr_permits", "max_permits", "rate")

  ---局部变量:上次申请的时间

  local last_mill_second=cacheInfo[1];

  ---局部变量:之前的令牌数

  local curr_permits=tonumber(cacheInfo[2]);

  ---局部变量:桶的容量

  local max_permits=tonumber(cacheInfo[3]);

  ---局部变量:令牌的发放速率

  local rate=cacheInfo[4];

  ---局部变量:本次的令牌数

  local local_curr_permits=max_permits;

  if (type(last_mill_second) ~='boolean' and last_mill_second ~=nil) then

  --计算时间段内的令牌数

  local reverse_permits=math.floor(((curr_mill_second - last_mill_second) / 1000) *rate);

  --令牌总数

  local expect_curr_permits=reverse_permits + curr_permits;

  --可以申请的令牌总数

  local_curr_permits=math.min(expect_curr_permits, max_permits);

  else

  --第一次获取令牌

  redis.pcall("HSET", key, "last_mill_second", curr_mill_second)

  end

  local result=-1;

  --有足够的令牌可以申请

  if (local_curr_permits - apply >=0) then

  --保存剩余的令牌

  redis.pcall("HSET", key, "curr_permits", local_curr_permits - apply);

  --保存时间,下次令牌获取时使用

  redis.pcall("HSET", key, "last_mill_second", curr_mill_second)

  --返回令牌获取成功

  result=1;

  else

  --保存令牌总数

  redis.pcall("HSET", key, "curr_permits", local_curr_permits);

  --返回令牌获取失败

  result=-1;

  end

  return result

  end

  ---方法:初始化限流器

  ---1 success

  ---@param key key

  ---@param max_permits 桶的容量

  ---@param rate 令牌的发放速率

  local function init(key, max_permits, rate)

  local rate_limit_info=redis.pcall("HMGET", key, "last_mill_second", "curr_permits", "max_permits", "rate")

  local org_max_permits=tonumber(rate_limit_info[3])

  local org_rate=rate_limit_info[4]

  if (org_max_permits==nil) or (rate ~=org_rate or max_permits ~=org_max_permits) then

  redis.pcall("HMSET", key, "max_permits", max_permits, "rate", rate, "curr_permits", max_permits)

  end

  return 1;

  end

  ---方法:删除限流Key

  local function delete(key)

  redis.pcall("DEL", key) return 1;

  end

  local key=KEYS[1]

  local method=ARGV[1]

  if method=='acquire' then

  return acquire(key, ARGV[2], ARGV[3])

  elseif method=='init' then

  return init(key, ARGV[2], ARGV[3])

  elseif method=='delete' then

  return delete(key)

  else

  --ignore

  end

  该脚本有3个方法,其中两个方法比较重要,分别说明如下:

  (1)限流器初始化方法init(key,max_permits,rate),此方法在限流开始时被调用。

  (2)限流检测的方法acquire(key,apply),此方法在请求到来时被调用。

  Java分布式令牌桶限流

  rate_limiter.lua脚本既可以在Java中调用,又可以在Nginx中调用。本小节先介绍其在Java中的使用,第10章再介绍其在Nginx中的使用。

  Java分布式令牌桶限流器的实现就是通过Java代码向Redis加载rate_limiter.lua脚本,然后封装其令牌桶初始化方法init(...)和限流监测方法acquire(...),以供外部调用。它的代码如下:

  package com.crazymaker.springcloud.standard.ratelimit;

  ...

  /**

  *实现:令牌桶限流服务

  *create by尼恩 @ 富贵论坛

  **/

  @Slf4j

  public class RedisRateLimitImpl implements RateLimitService, InitializingBean

  {

  /**

  *限流器的redis key前缀

  */

  private static final String RATE_LIMITER_KEY_PREFIX="rate_limiter:";

  //private ScheduledExecutorService executorService=Executors.newScheduledThreadPool(1);

  private RedisRateLimitProperties redisRateLimitProperties;

  private RedisTemplate redisTemplate;

  //lua脚本的实例

  private static RedisScript rateLimiterScript=null;

  //lua脚本的类路径

  private static String rateLimitLua="script/rate_limiter.lua";

  static

  {

  //从类路径文件中加载令牌桶lua脚本

  String script=IOUtil.loadJarFile(RedisRateLimitImpl.class.getClassLoader(), rateLimitLua);

  if (StringUtils.isEmpty(script))

  {

  log.error("lua script load failed:" + rateLimitLua);

  } else

  {

  //创建Lua脚本实例

  rateLimiterScript=new DefaultRedisScript<>(script, Long.class);

  }

  }

  public RedisRateLimitImpl(

  RedisRateLimitProperties redisRateLimitProperties,

  RedisTemplate redisTemplate)

  {

  thisisRateLimitProperties=redisRateLimitProperties;

  thisisTemplate=redisTemplate;

  }

  private Map<String, LimiterInfo> limiterInfoMap=new HashMap<>();

  /**

  *限流器的信息

  */

  @Builder

  @Data

  public static class LimiterInfo

  {

  /**

  *限流器的key,如秒杀的id

  */ private String key;

  /**

  *限流器的类型,如seckill

  */

  private String type="default";

  /**

  *限流器的最大桶容量

  */

  private Integer maxPermits;

  /**

  *限流器的速率

  */

  private Integer rate;

  /**

  *限流器的redis key

  */

  public String fullKey()

  {

  return RATE_LIMITER_KEY_PREFIX + type + ":" + key;

  }

  /**

  *限流器在map中的缓存key

  */

  public String cashKey()

  {

  return type + ":" + key;

  }

  }

  /**

  *限流检测:是否超过redis令牌桶限速器的限制

  *

  *@param cacheKey计数器的key

  *@return true or false

  */

  @Override

  public Boolean tryAcquire(String cacheKey)

  {

  if (cacheKey==null)

  {

  return true;

  }

  if (cacheKey.indexOf(":") <=0)

  {

  cacheKey="default:" + cacheKey;

  }

  LimiterInfo limiterInfo=limiterInfoMap.get(cacheKey);

  if (limiterInfo==null)

  {

  return true;

  }

  Long acquire=(Long) redisTemplate.execute(rateLimiterScript,

  ImmutableList.of(limiterInfo.fullKey()),

  "acquire",

  "1");

  if (acquire==1)

  {

  return false;

  }

  return true;

  }

  /**

  *重载方法:限流器初始化

  *

  *@param limiterInfo限流的类型

  */

  public void initLimitKey(LimiterInfo limiterInfo)

  {

  if (null==rateLimiterScript)

  {

  return;

  }

  String maxPermits=limiterInfo.getMaxPermits().toString();

  String rate=limiterInfo.getRate().toString();

  //执行redis脚本

  Long result=(Long) redisTemplate.execute(rateLimiterScript,

  ImmutableList.of(limiterInfo.fullKey()),

  "init",

  maxPermits,

  rate); limiterInfoMap.put(limiterInfo.cashKey(), limiterInfo);

  }

  /**

  *限流器初始化

  *

  *@param type类型

  *@param key id

  *@param maxPermits上限

  *@param rate 速度

  */

  public void initLimitKey(String type, String key,

  Integer maxPermits, Integer rate)

  {

  LimiterInfo limiterInfo=LimiterInfo.builder()

  .type(type)

  .key(key)

  .maxPermits(maxPermits)

  .rate(rate)

  .build();

  initLimitKey(limiterInfo);

  }

  /**

  *获取redis lua脚本的sha1编码,并缓存到redis

  */

  public String cacheSha1()

  {

  String sha1=rateLimiterScript.getSha1();

  redisTemplate.opsForValue().set("lua:sha1:rate_limiter", sha1);

  return sha1;

  }

  }Java分布式令牌桶限流的自验证

  自验证的工作:首先初始化分布式令牌桶限流器,然后使用两条

  线程不断进行限流的检测。自验证的代码如下:

  package com.crazymaker.springcloud.ratelimit;

  ...

  @Slf4j

  @RunWith(SpringRunner.class)

  //指定启动类

  @SpringBootTest(classes={DemoCloudApplication.class})

  /**

  *redis分布式令牌桶测试类

  */

  public class RedisRateLimitTest

  {

  @Resource(name="redisRateLimitImpl")

  RedisRateLimitImpl limitService;

  //线程池,用于多线程模拟测试

  private ExecutorService pool=Executors.newFixedThreadPool(10);

  @Test

  public void testRedisRateLimit()

  {

  //初始化分布式令牌桶限流器

  limitService.initLimitKey(

  "seckill", //redis key中的类型

  "10000", //redis key中的业务key,比如商品id

  2, //桶容量

  2); //每秒令牌数

  AtomicInteger count=new AtomicInteger();

  long start=System.currentTimeMillis();

  //线程数

  final int threads=2;

  //每条线程的执行轮数

  final int turns=20;

  //同步器

  CountDownLatch countDownLatch=new CountDownLatch(threads);

  for (int i=0; i < threads; i++)

  {

  pool.submit(() ->

  {

  try

  {

  //每个用户访问turns次

  for (int j=0; j < turns; j++)

  {

  boolean limited=limitService.tryAcquire

  ("seckill:10000");

  if (limited)

  {

  count.getAndIncrement();

  }

  Thread.sleep(200);

  }

  } catch (Exception e)

  { e.printStackTrace();

  }

  countDownLatch.countDown();

  });

  }

  try

  {

  countDownLatch.await();

  } catch (InterruptedException e)

  {

  e.printStackTrace();

  }

  float time=(System.currentTimeMillis() - start) / 1000F;

  //输出统计结果

  log("限制的次数为:" + count.get() + " 时长为:" + time);

  log("限制的次数为:" + count.get() +

  ",通过的次数为:" + (threads *turns - count.get()));

  log("限制的大专比例为:" +

  (float) count.get() / (float) (threads *turns));

  log("运行的时长为:" + time);

  try

  {

  Thread.sleep(Integer.MAX_VALUE);

  } catch (InterruptedException e)

  {

  e.printStackTrace();

  }

  }

  }

  两条线程各运行20次,每一次运行休眠200毫秒,总计耗时4秒,

  运行40次,部分输出结果如下:

  [main] INFO c.c.s.risRateLimitTest - 限制的次数为:32 时长为:4.015

  [main] INFO c.c.s.risRateLimitTest - 限制的次数为:32,通过的次数为:8

  [main] INFO c.c.s.risRateLimitTest - 限制的比例为:0.8

  [main] INFO c.c.s.risRateLimitTest - 运行的时长为:4.015

  大家可以自行调整参数,运行以上自验证程序并观察实验结果,体验一下分布式令牌桶限流的效果。

  本文给大家讲解的内容是高并发核心编程,限流原理与实战,实战:分布式令牌桶限流下篇文章给大家讲解的是高并发核心编程,Spring Cloud+Nginx秒杀实战;觉得文章不错的朋友可以转发此文关注小编;感谢大家的支持!

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
30天前
|
数据管理 API 调度
鸿蒙HarmonyOS应用开发 | 探索 HarmonyOS Next-从开发到实战掌握 HarmonyOS Next 的分布式能力
HarmonyOS Next 是华为新一代操作系统,专注于分布式技术的深度应用与生态融合。本文通过技术特点、应用场景及实战案例,全面解析其核心技术架构与开发流程。重点介绍分布式软总线2.0、数据管理、任务调度等升级特性,并提供基于 ArkTS 的原生开发支持。通过开发跨设备协同音乐播放应用,展示分布式能力的实际应用,涵盖项目配置、主界面设计、分布式服务实现及部署调试步骤。此外,深入分析分布式数据同步原理、任务调度优化及常见问题解决方案,帮助开发者掌握 HarmonyOS Next 的核心技术和实战技巧。
202 76
鸿蒙HarmonyOS应用开发 | 探索 HarmonyOS Next-从开发到实战掌握 HarmonyOS Next 的分布式能力
|
30天前
|
物联网 调度 vr&ar
鸿蒙HarmonyOS应用开发 |鸿蒙技术分享HarmonyOS Next 深度解析:分布式能力与跨设备协作实战
鸿蒙技术分享:HarmonyOS Next 深度解析 随着万物互联时代的到来,华为发布的 HarmonyOS Next 在技术架构和生态体验上实现了重大升级。本文从技术架构、生态优势和开发实践三方面深入探讨其特点,并通过跨设备笔记应用实战案例,展示其强大的分布式能力和多设备协作功能。核心亮点包括新一代微内核架构、统一开发语言 ArkTS 和多模态交互支持。开发者可借助 DevEco Studio 4.0 快速上手,体验高效、灵活的开发过程。 239个字符
216 13
鸿蒙HarmonyOS应用开发 |鸿蒙技术分享HarmonyOS Next 深度解析:分布式能力与跨设备协作实战
|
1月前
|
NoSQL Java Redis
秒杀抢购场景下实战JVM级别锁与分布式锁
在电商系统中,秒杀抢购活动是一种常见的营销手段。它通过设定极低的价格和有限的商品数量,吸引大量用户在特定时间点抢购,从而迅速增加销量、提升品牌曝光度和用户活跃度。然而,这种活动也对系统的性能和稳定性提出了极高的要求。特别是在秒杀开始的瞬间,系统需要处理海量的并发请求,同时确保数据的准确性和一致性。 为了解决这些问题,系统开发者们引入了锁机制。锁机制是一种用于控制对共享资源的并发访问的技术,它能够确保在同一时间只有一个进程或线程能够操作某个资源,从而避免数据不一致或冲突。在秒杀抢购场景下,锁机制显得尤为重要,它能够保证商品库存的扣减操作是原子性的,避免出现超卖或数据不一致的情况。
63 10
|
1月前
|
缓存 NoSQL Java
高并发场景秒杀抢购超卖Bug实战重现
在电商平台的秒杀活动中,高并发场景下的抢购超卖Bug是一个常见且棘手的问题。一旦处理不当,不仅会引发用户投诉,还会对商家的信誉和利益造成严重损害。本文将详细介绍秒杀抢购超卖Bug的背景历史、业务场景、底层原理以及Java代码实现,旨在帮助开发者更好地理解和解决这一问题。
74 12
|
2月前
|
存储 Dubbo Java
分布式 RPC 底层原理详解,看这篇就够了!
本文详解分布式RPC的底层原理与系统设计,大厂面试高频,建议收藏。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
分布式 RPC 底层原理详解,看这篇就够了!
|
1月前
|
机器学习/深度学习 存储 运维
分布式机器学习系统:设计原理、优化策略与实践经验
本文详细探讨了分布式机器学习系统的发展现状与挑战,重点分析了数据并行、模型并行等核心训练范式,以及参数服务器、优化器等关键组件的设计与实现。文章还深入讨论了混合精度训练、梯度累积、ZeRO优化器等高级特性,旨在提供一套全面的技术解决方案,以应对超大规模模型训练中的计算、存储及通信挑战。
84 4
|
3月前
|
NoSQL Java Redis
开发实战:使用Redisson实现分布式延时消息,订单30分钟关闭的另外一种实现!
本文详细介绍了 Redisson 延迟队列(DelayedQueue)的实现原理,包括基本使用、内部数据结构、基本流程、发送和获取延时消息以及初始化延时队列等内容。文章通过代码示例和流程图,逐步解析了延迟消息的发送、接收及处理机制,帮助读者深入了解 Redisson 延迟队列的工作原理。
|
3月前
|
NoSQL Java Redis
太惨痛: Redis 分布式锁 5个大坑,又大又深, 如何才能 避开 ?
Redis分布式锁在高并发场景下是重要的技术手段,但其实现过程中常遇到五大深坑:**原子性问题**、**连接耗尽问题**、**锁过期问题**、**锁失效问题**以及**锁分段问题**。这些问题不仅影响系统的稳定性和性能,还可能导致数据不一致。尼恩在实际项目中总结了这些坑,并提供了详细的解决方案,包括使用Lua脚本保证原子性、设置合理的锁过期时间和使用看门狗机制、以及通过锁分段提升性能。这些经验和技巧对面试和实际开发都有很大帮助,值得深入学习和实践。
太惨痛: Redis 分布式锁 5个大坑,又大又深, 如何才能 避开 ?
|
1月前
|
存储 NoSQL Java
使用lock4j-redis-template-spring-boot-starter实现redis分布式锁
通过使用 `lock4j-redis-template-spring-boot-starter`,我们可以轻松实现 Redis 分布式锁,从而解决分布式系统中多个实例并发访问共享资源的问题。合理配置和使用分布式锁,可以有效提高系统的稳定性和数据的一致性。希望本文对你在实际项目中使用 Redis 分布式锁有所帮助。
163 5
|
2月前
|
NoSQL Java 数据处理
基于Redis海量数据场景分布式ID架构实践
【11月更文挑战第30天】在现代分布式系统中,生成全局唯一的ID是一个常见且重要的需求。在微服务架构中,各个服务可能需要生成唯一标识符,如用户ID、订单ID等。传统的自增ID已经无法满足在集群环境下保持唯一性的要求,而分布式ID解决方案能够确保即使在多个实例间也能生成全局唯一的标识符。本文将深入探讨如何利用Redis实现分布式ID生成,并通过Java语言展示多个示例,同时分析每个实践方案的优缺点。
87 8