作者:神牛003

来源:https://www.cnblogs.com/wangrudong003/p/10627539.html

本篇内容主要讲解的是redis分布式锁,这个在各大厂面试几乎都是必备的,下面结合模拟抢单的场景来使用她;本篇不涉及到的redis环境搭建,快速搭建个人测试环境,这里建议使用docker;本篇内容节点如下:

jedis的nx生成锁

  • 如何删除锁
  • 模拟抢单动作(10w个人开抢)
  • jedis的nx生成锁

对于java中想操作redis,好的方式是使用jedis,首先pom中引入依赖:

    redis.clients    jedis

对于分布式锁的生成通常需要注意如下几个方面:

  • 创建锁的策略:redis的普通key一般都允许覆盖,A用户set某个key后,B在set相同的key时同样能成功,如果是锁场景,那就无法知道到底是哪个用户set成功的;这里jedis的setnx方式为我们解决了这个问题,简单原理是:当A用户先set成功了,那B用户set的时候就返回失败,满足了某个时间点只允许一个用户拿到锁。
  • 锁过期时间:某个抢购场景时候,如果没有过期的概念,当A用户生成了锁,但是后面的流程被阻塞了一直无法释放锁,那其他用户此时获取锁就会一直失败,无法完成抢购的活动;当然正常情况一般都不会阻塞,A用户流程会正常释放锁;过期时间只是为了更有保障。

下面来上段setnx操作的代码:

这里注意点在于jedis的set方法,其参数的说明如:

  • NX:是否存在key,存在就不set成功
  • PX:key过期时间单位设置为毫秒(EX:单位秒)

setnx如果失败直接封装返回false即可,下面我们通过一个get方式的api来调用下这个setnx方法:

@GetMapping("/setnx/{key}/{val}")public boolean setnx(@PathVariable String key, @PathVariable String val) {     return jedisCom.setnx(key, val);}

访问如下测试url,正常来说第一次返回了true,第二次返回了false,由于第二次请求的时候redis的key已存在,所以无法set成功

由上图能够看到只有一次set成功,并key具有一个有效时间,此时已到达了分布式锁的条件。

如何删除锁

上面是创建锁,同样的具有有效时间,但是我们不能完全依赖这个有效时间,场景如:有效时间设置1分钟,本身用户A获取锁后,没遇到什么特殊情况正常生成了抢购订单后,此时其他用户应该能正常下单了才对,但是由于有个1分钟后锁才能自动释放,那其他用户在这1分钟无法正常下单(因为锁还是A用户的),因此我们需要A用户操作完后,主动去解锁:

这里也使用了jedis方式,直接执行lua脚本:根据val判断其是否存在,如果存在就del;

其实个人认为通过jedis的get方式获取val后,然后再比较value是否是当前持有锁的用户,如果是那最后再删除,效果其实相当;只不过直接通过eval执行脚本,这样避免多一次操作了redis而已,缩短了原子操作的间隔。(如有不同见解请留言探讨);同样这里创建个get方式的api来测试:

@GetMapping("/delnx/{key}/{val}")public int delnx(@PathVariable String key, @PathVariable String val) {   return jedisCom.delnx(key, val);}

注意的是delnx时,需要传递创建锁时的value,因为通过et的value与delnx的value来判断是否是持有锁的操作请求,只有value一样才允许del;

模拟抢单动作(10w个人开抢)

有了上面对分布式锁的粗略基础,我们模拟下10w人抢单的场景,其实就是一个并发操作请求而已,由于环境有限,只能如此测试;如下初始化10w个用户,并初始化库存,商品等信息,如下代码:

有了上面10w个不同用户,我们设定商品只有10个库存,然后通过并行流的方式来模拟抢购,如下抢购的实现:

这里实现的逻辑是:

  • parallelStream():并行流模拟多用户抢购
  • (startTime + timeout) >= System.currentTimeMillis():判断未抢成功的用户,timeout秒内继续获取锁
  • 获取锁前和后都判断库存是否还足够
  • jedisCom.setnx(shangpingKey, b):用户获取抢购锁
  • 获取锁后并下单成功,最后释放锁:jedisCom.delnx(shangpingKey, b)

再来看下记录的日志结果:

最终返回抢购成功的用户:

redis set时间单位_SpringBoot+Redis分布式锁:模拟抢单相关推荐

  1. java如何保证redis设置过期时间的原子性_分布式锁用 Redis 还是 Zookeeper

    在讨论这个问题之前,我们先来看一个业务场景: 系统A是一个电商系统,目前是一台机器部署,系统中有一个用户下订单的接口,但是用户下订单之前一定要去检查一下库存,确保库存足够了才会给用户下单. 由于系统有 ...

  2. 【Redis Lua 脚本 可重入分布式锁】

    文章目录 前言 一.最简单的版本:setnx key value 获取锁成功 获取锁失败 释放锁 缺点 二.升级版本:set key value [ex seconds] [nx] 获取锁成功 获取锁 ...

  3. 视频教程- 19年录制Redis实战教程 高可用秒杀分布式锁布隆过滤器实战 SpringBoot教程整合-Java

    19年录制Redis实战教程 高可用秒杀分布式锁布隆过滤器实战 SpringBoot教程整合 7年的开发架构经验,曾就职于国内一线互联网公司,开发工程师,现在是某创业公司技术负责人, 擅长语言有nod ...

  4. Redis 作者 Antirez 讲如何实现分布式锁?Redis 实现分布式锁天然的缺陷分析Redis分布式锁的正确使用姿势!...

    Redis分布式锁基本原理 采用 redis 实现分布式锁,主要是利用其单线程命令执行的特性,一般是 setnx, 只会有一个线程会执行成功,也就是只有一个线程能成功获取锁:看着很完美. 然而-- 看 ...

  5. redis 的惊群处理和分布式锁的应用例子

    在并发量比较高的情况下redis有很多应用场景,提升查询效率,缓解底层DBio ,下面列举两个平时开发中应用过的两个例子,欢迎各位一起讨论改进. 1 . redis 惊群处理 1.1 方案的由来 Re ...

  6. redis mysql 解决超卖_Redis 分布式锁解决超卖问题

    Redis 分布式锁解决超卖问题 1,Redis 事物介绍 1. Redis 事物是可以一次执行多个命令, 本质是一组命令的集合. 2. 一个事务中的所有命令都会序列化, 按顺序串行化的执行而不会被其 ...

  7. 什么是分布式锁?redis、zookeeper、etcd实现分布式锁有什么不同之处?

    目录 分布式锁定义 目的 基于redis分布式锁 基于zookeeper实现的分布式锁 edis.zookeeper.etcd实现分布式锁的比较 建议选择etcd实现分布式锁 分布式锁定义 分布式环境 ...

  8. redis 3.0 java 工具包_redis分布式锁工具包,提供纯Java方式调用

    redis-distributed-lock redis分布式锁工具包,提供纯Java方式调用,支持传统Spring工程, 为spring boot应用提供了starter,更方便快捷的调用. 项目结 ...

  9. redis实现setnx,setex连用实现分布式锁

    redis实现分布式锁 1.主要命令: setnx setex 2.主要问题: 使用redis实现分布式锁,利用上面两个命令的特性.但是最重要的是锁要有过期时间,不然万一服务器宕机或者redis宕机, ...

最新文章

  1. java super extends_Java继承extends与super关键字
  2. 20个优秀的移动(iPhone)网站设计案例
  3. datatables 响应式
  4. ANTLR-语法树遍历机制
  5. 人工智能学习该读哪些书籍?
  6. undefined reference to `create_module''
  7. Sybase数据库故障的修复方法
  8. 操作系统银行家算法(课程设计报告)JAVA实现
  9. javascript之函数的定义传参
  10. Excel-缺失数据处理
  11. i春秋《从0到1:CTFer成长之路》通关WP
  12. STM32串口IAP
  13. pmp项目管理的优先级解决方法
  14. Boson Netsim 思科路由器仿真 静态路由表、RIP、OSPF
  15. 3D MAX界面操作教程及渲染技巧
  16. 浙江省2021年高新技术企业认定补贴奖励查询及优惠政策,国高达60万
  17. css 按钮扩散效果,专治按钮效果不明显(扩散动画效果)
  18. Windows安全机制——UAC(用户权限控制)
  19. python写wordcount_Python开发Spark应用之Wordcount词频统计
  20. 电动汽车驱动防滑控制系统设计

热门文章

  1. java checkbox数组_Java中的复选框数组
  2. Python案例:输出指定范围的闰年
  3. 16软件1班安卓第一学期学习总结
  4. Python程序:输出斐波那契数列
  5. php输出mysql查询结果_PHP简单获取数据库查询结果并返回JSON
  6. bzoj1211 [HNOI2004]树的计数 prufer序列+组合数
  7. 基于STM32实现的普通灯泡调光控制器
  8. 屏幕空间环境光遮蔽(SSAO)算法的实现
  9. access用扫描枪输入_判断是否扫码枪输入的通用函数源码
  10. abrels.inc.php_fckk.php