【Redis】应用一 : 分布式锁

一、分布式锁

本质就是实现一个线程在 Redis 中占坑, 用完了,再 del 释放坑

问题1: 程序出现异常, 导致 del 指令未被调用

解决1: 设置一个过期时间

问题2:expire 得不到执行,导致死锁

在这里插入图片描述

解决2:

最初:使用 Redis 社区提供分布式锁

最终: Redis 2.8 版本中加入了 set 指令的扩展参数,使得 setnx 和 expire 指令可以一起执行

set lock:codehole true ex 5 

问题3:超时问题,未执行完,就释放锁了

解决3: 较长时间的任务,不用 Redis 分布式锁

为 set 指令的 value 设置随机数,释放锁时匹配随机数是否一致 (匹配和删除不是原子操作,需要使用 Lua 脚本)

if redis.call("get",KEYS[1]) == ARGV[1] then
 return redis.call("del",KEYS[1])
else
 return 0
end

二、可重入性

可重入性:线程在持有锁的情况下再次请求加锁( ReentrantLock)。

Redis 分布式锁实现可重入 : 对客户端的 set 方法进行包装,使用线程的 Threadlocal 变量存储当前持有锁的计数。

不推荐在客户端实现使用可重入锁,它加重了客户端的复杂性

在编写业务方法时注意在逻辑结构上进行调整就可以,如下:


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用Redis实现分布式锁的具体步骤如下: 1. 使用Redis的SETNX命令获取锁。SETNX的作用是设置一个键值对,如果该键不存在,则设置成功并返回1,否则设置失败并返回0。 2. 如果SETNX命令返回1,则表示获取锁成功,可以执行业务代码;否则设置失败,需要等待一段时间后重试。 3. 在业务代码执行完成后,使用Redis的DEL命令释放锁。 4. 为了避免死锁问题,需要设置锁的超时时间,超时时间到了之后锁会自动释放。可以使用Redis的EXPIRE命令来设置锁的超时时间。 下面是一个使用Redis实现分布式锁的示例代码: ``` import redis import time class RedisLock: def __init__(self, redis_client, lock_name, acquire_timeout=10, expire_time=10): self.redis_client = redis_client self.lock_name = f"lock:{lock_name}" self.acquire_timeout = acquire_timeout self.expire_time = expire_time def acquire(self): start_time = time.time() while time.time() - start_time < self.acquire_timeout: if self.redis_client.setnx(self.lock_name, 1): self.redis_client.expire(self.lock_name, self.expire_time) return True time.sleep(0.1) return False def release(self): self.redis_client.delete(self.lock_name) ``` 在使用时,可以创建一个RedisLock对象,然后调用acquire方法获取锁,执行业务代码,最后调用release方法释放锁。 ``` redis_client = redis.Redis(host='localhost', port=6379, db=0) lock = RedisLock(redis_client, 'my_lock') if lock.acquire(): try: # 执行业务代码 pass finally: lock.release() else: print('获取锁失败') ``` 需要注意的是,上面的示例代码只是一个简单的实现,实际应用中需要根据具体需求进行优化和改进。比如,可以使用RedLock算法来解决Redis的单点故障问题,提高分布式锁的稳定性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值