redis分布式锁

本文深入探讨Redis分布式锁的实现机制,包括使用setnx命令解决并发加锁问题,通过设置过期时间防止死锁,利用Lua脚本确保锁操作的原子性,以及在集群环境下采用RedLock算法保证锁的一致性和可靠性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

原文:https://baijiahao.baidu.com/s?id=1623086259657780069&wfr=spider&for=pc

redis 分布式锁的实现方法,表面上看很简单。
指定一个key, 获取锁 ,查询一下redis中 这个key有没有值,没有,就设置,获取锁就成功了,
释放锁,就删除这个key

  1. 问题:并发判断redis中 都没有key, 同时去加锁,会出现覆盖问题
    解决: 使用 setnx 命令,不存在才设置,只有一个能设置锁成功
  2. 问题: 线程还未来得及释放锁,程序奔溃了,其他的都被阻塞了。
    解决: 对加入的锁,设置一个过期时间。
  3. 问题: setnx 和 expire 添加和设置过期不是原子操作,还是会出现锁无法释放
    解决: redis2.8以后新增了指令 set lock:key true ex 5 nx
    使用LUA脚本设置,是比较通用的方式
String script = "local rs=redis.call('setnx',KEYS[1],ARGV[1]);if(rs<1) then return 'F';end;redis.call('expire',KEYS[1],tonumber(ARGV[2]));return 'S';";
Jedis jedis = shardedJedis.getShard(redisKey);  
Object result = jedis.evalsha(script, keys, args);
if ("S".equals(result)) {
    return true;
}
  出处:https://www.2cto.com/kf/201809/780811.html
  1. 问题: 释放锁的时候,锁过期,被其他线程获取到锁,然后这个线程执行后释放了锁
    解决: 加锁的时候设置一个随机值,释放锁的时候,先取出来判断 这个值是否有变更,如果变更了说明锁已经被其他人占有了,没有变更,释放锁。 这个地方 判断和 释放锁,不是原子性,需要使用lua脚本 保证原子性

以上都针对的是 单机的情况。
如果 有备机切换的话,这种情况就有问题了。
那怎么办?

redis集群的分布式锁

Redisson 实现方式(红锁 RedLock)

github Redisson https://github.com/redisson/redisson

 <dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
   <version>3.9.0</version>
</dependency>  

<!-- JDK 1.6+ compatible -->
<dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
   <version>2.14.0</version>
</dependency>

RedLock 实现的算法规则

  1. 加锁时,给redis集群的 每一个master 都加锁。
  2. 加锁成功的条件:a. 超过半数的机器加锁成功,b. 总加锁时间,小于单个锁超时时间
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值