基于Redis实现分布式锁

本文介绍了Redis中实现分布式锁的关键命令SETNX与GETSET,并探讨了如何通过这些命令保证加锁和释放锁的原子性,避免死锁问题。此外,文章还讨论了分布式锁的可靠性需求及其基于Lua脚本的实现。

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

相关命令

1) SETNX(SET if Not eXists)

将 key 的值设为 value ,当且仅当 key 不存在。

若给定的 key 已经存在,则 SETNX 不做任何动作。

SETNX 是『SET if Not eXists』(如果不存在,则 SET)的简写

返回值:
  设置成功,返回 1 。
  设置失败,返回 0 。

 

redis官网有一篇文章专门谈论了实现分布式锁的话题。基本的原则是:采用setnx尝试获取锁并判断是否获得了锁,setnx设置的值是它想占用锁的时间(预估)

 

2) GETSET

将给定 key 的值设为 value ,并返回 key 的旧值(old value)。

  当 key 存在但不是字符串类型时,返回一个错误。

返回值:
  返回给定 key 的旧值。
  当 key 没有旧值时,也即是, key 不存在时,返回 nil 。

 

可靠性

首先,为了确保分布式锁可用,我们至少要确保锁的实现同时满足以下四个条件:

  1. 互斥性。在任意时刻,只有一个客户端能持有锁。
  2. 不会发生死锁。即使有一个客户端在持有锁的期间崩溃而没有主动解锁,也能保证后续其他客户端能加锁。
  3. 具有容错性。只要大部分的Redis节点正常运行,客户端就可以加锁和解锁。
  4. 解铃还须系铃人。加锁和解锁必须是同一个客户端,客户端自己不能把别人加的锁给解了。

 

加锁

重试机制

 

保证加锁的原子性

加锁操作和设置过期时间操作能够整体成功或失败,需要保证原子性

 

 

 

释放锁

保证释放锁的原子性

锁需要有标识,不能出现以下的情况:

  • 线程T1获取锁
  • 线程T1执行业务操作,由于某些原因阻塞了较长时间
  • 锁自动过期,即锁自动释放了
  • 线程T2获取锁
  • 线程T1业务操作完毕,释放锁(其实是释放的线程T2的锁)

 

使用LUA脚本释放锁

 

 

基于AOP 的redis分布式锁

 

参考

Redis 分布式锁的正确实现方式( Java 版 ) 好文

手把手教你实现一个基于Redis的分布式锁

spring boot redis分布式锁

一个Redis实现的分布式锁

 

转载于:https://www.cnblogs.com/tonyq/p/9225198.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值