##学习笔记 Redis分布式锁(高并发场景)

学习内容来自于腾讯课堂 图灵老师

在这里插入图片描述
在这里插入图片描述

stringRedisTemplate.opsForValue().get(‘key’) 通过key获取redis库存数量时,在高并发场景下会出现同一个库存被多次获取的情况。而synchronized锁是进程锁,只针对同一个jvm情况下,也就是在分布式,或者说相同war部署了多个Tomcat的时候,synchronized锁并不能解决以上库存被多次获取的情况。这里有个疑问就是redis是线程安全的,或者说他是单线程的,怎么会出现库存被多少获取呢,因为在业务代码上,一般首先我们是先通过stringRedisTemplate.opsForValue().get(‘key’)获取值,再通过stringRedisTemplate.opsForValue().set(‘key’,value)修改原先的值,而在高并发的时候,获取redis的key值时,同一个库存会被多次获取。
此时,通过stringRedisTemplate.opsForValue().setIfAbsent(‘key’,value)(注:这是springBoot集成Redis的,以上stringRedisTemplate都是。)实现redis分布式锁,想当于redis的setnx方法。方法含义分别如下:
redis命令
在这里插入图片描述

stringRedisTemplate命令

在这里插入图片描述

,这里又因为redis是线程安全的,所以在设置stringRedisTemplate.opsForValue().setIfAbsent (‘key’,value)时,高并发场景下已队列的形式在等待,假如key不存在的话,返回true,我们使业务往下继续执行,假如key存在的话,则结束这块业务。
下面新的问题来了,当业务块发生异常时,要释放库存,我们需要在finally中关闭锁,如下
在这里插入图片描述

另一个场景便是我们要考虑,当服务器宕机或者说当运维重启服务器时,此时刚好创建的redis分布式锁,但是下面的业务没有实现的情况下,需要释放锁(注:其他业务涉及数据库,可开启事务),如下,我们需要给redis库对应的值通过stringRedisTemplate.opsForValue().setIfAbsent (‘key’,value,time,timeType)设置失效时间
在这里插入图片描述

再一个场景是,一般上我们设置分布式锁的key,是通过由前端的传来的商品主键ID,作为key值,在高并发场景下,必然会出现相同的key值,而我们设置key值的有效时间不足,代码块业务还未走完,比如我们设置了失效时间为10秒,但是代码最终走到finally中时花了15秒时间,此时第二个线程相同的key也许才刚刚创建,就被第一个线程中的finally中stringRedisTemplate.delete(‘key’)方法清理了。这样导致第二个线程或者其他线程的分布锁失效,所以我们要给对应的分步锁添加唯一value值。如下

在这里插入图片描述

另Redisson实现方式
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在高并发下,可以使用 Redis 的 SETNX 命令来实现分布式。SETNX 指令是将 key 的值设为 value,当且仅当 key 不存在。 步骤如下: 1. 使用 SETNX 命令尝试获取,如果返回值为 1,则说明获取成功。 2. 使用 EXPIRE 命令为设置过期时间,防止死。 3. 在业务代码执行完成后,使用 DEL 命令释放。 为了防止网络延迟等问题,可以在获取时设置一个随机值,并在释放时判断该值是否与当前对应的值相同,以确保只有持有的客户端才能释放### 回答2: 高并发下Redis分布式的实现是通过Redis的setnx命令和expire命令来实现的。 首先,通过setnx命令尝试设置一个带有过期时间的key,如果成功设置,则表示获取到了分布式。如果设置失败,说明已经被其他客户端占用,需要等待或进行重试。 为了防止因为某个客户端处理时间过长而导致过期的情况,可以为的过期时间设置一个合理的值。在获取到后,可以使用expire命令为的key设置过期时间,确保在一定时间内释放。 为了提高的安全性,可以为每个客户端设置一个唯一的ID作为的值,并将名称与该ID进行绑定。这样可以确保只有获取的客户端才能释放,防止其他客户端误释放。 另外,考虑到高并发情况下的竞争,可以在获取失败后进行等待一段时间再进行重试,避免频繁的竞争对系统性能造成负面影响。 需要注意的是,Redis分布式的实现并不能解决所有并发问题,仅适用于单个业务场景下的加和释放操作。在设计和使用时需要考虑到具体业务需求和场景,并进行适当的优化和调整。 ### 回答3: 在高并发场景下,为了保证数据的一致性和并发执行的正确性,常常需要使用分布式来控制对共享资源的访问。Redis作为一个高性能的内存键值存储系统,也可以用来实现分布式Redis实现分布式的一种常见方式是使用SETNX命令。当多个客户端同时尝试获取时,只有一个客户端能成功执行SETNX操作,即将对应的key设置为1,表示获取了。其他客户端获取失败,需要等待被释放后重新尝试。 为了保证的正确性和防止死,还需要为设置一个合理的过期时间,以防止获取的客户端因为异常情况导致无法及时释放。 在高并发环境下,为了提高的性能,可以考虑使用红机制。红是将分布式Redis的复制功能相结合,确保在大部分节点上都获取到后,才认为已经获得。这样可以避免某个节点出现故障或网络异常导致丢失的情况。 另外,为了避免因为程序异常导致无法释放的情况,可以在获取到之后,使用Lua脚本来保证在一个原子操作中判断的状态并释放。 在实现分布式时,还需要考虑高并发下的性能问题。可以通过优化Redis的部署结构、增加Redis的内存和CPU资源,以及使用连接池等方法来提高Redis的性能和并发能力,从而提高分布式的性能。 综上所述,高并发下的Redis分布式的实现主要包括使用SETNX命令获取、设置合理的超时时间、使用红机制增强的可靠性、使用Lua脚本保证原子操作、优化Redis的性能和并发能力等方面。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值