Redis 分布式锁遇到的序列化问题,细致入微全是精华!

本文详细记录了一次在使用Redis实现分布式锁时遇到的释放锁失败的问题,原因是RedisTemplate和StringRedisTemplate的序列化方式不一致。通过排查代码和配置,最终确定问题并解决,确保了锁的正常释放。
摘要由CSDN通过智能技术生成

场景描述

最近使用 Redis 遇到了一个类似分布式锁的场景,跟 Redis 实现分布式锁类比一下,就是释放锁失败,也就是缓存删不掉。又踩了一个 Redis 的坑……

这是什么个情况、又是怎样排查的呢?

本文主要对此做个复盘。

问题排查

既然是释放锁有问题,那就先看看释放锁的代码吧。

释放锁

释放锁使用了 Lua 脚本,代码逻辑和 Lua 脚本如下:

  • 释放锁示例代码
public Object release(String key, String value) {  
  Object existedValue = stringRedisTemplate.opsForValue().get(key);  
  log.info("key:{}, value:{}, redis旧值:{}", key, value, existedValue);  
    
  DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>(COMPARE_AND_DELETE, Long.class);  
  return stringRedisTemplate.execute(redisScript, Collections.singletonList(key), value);  
}  

  • 释放锁使用的 Lua 脚本
if redis.call('get',KEYS[1]) == ARGV[1]  
then  
    return redis.call('del',KEYS[1])  
else  
    return 0  
end;  

删除脚本中,会先获取 Redis key 的旧值,并与入参 value 比较,二者相等时才会删除。

如果释放成功,也就是 Redis 缓存删除成功,返回值为 1,否则失败返回为 0。

乍一看代码似乎没啥问题,测一下试试?

不过既然要释放锁,在此之前肯定要加锁,先看看加锁的逻辑吧。

加锁

说到加锁这里的逻辑,代码里有两种实现方式:

  • 示例代码一
public Object lock01(String key, String value) {  
  log.info("lock01, key={}, value={}", key, value);  
  return redisTemplate.opsForValue().setIfAbsent(key, value, LOCKED_TIME, TimeUnit.SECONDS);  
}  

  • 示例代码二

                
  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

图灵课堂诸葛

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值