业务场景:
在上一篇-Java业务功能并发问题处理的最后,我们用RedisTemplate
实现了一个分布式锁,但是后面又有用户反馈同个单据出现了重复操作,让我们回忆下上次的加锁代码:
问题描述:
原因出现在我们锁住的那段代码执行了太久,超过预设的过期时长。在A线程还在执行中时,Redis
中作为锁的key已经过期了,当B线程进入时判断已经没有锁了,因此允许执行。
解决方案分析:
由于我们不知道业务需要执行多久,需要一个机制在过期前检查是否线程还在运行中(为什么必须是过期前?因为过期了我们就无法知道线程是超时导致的解锁还是线程主动的解锁。),如果线程仍在运行,则将过期时间延长。下面绘制一个流程图让大家更直观地了解整个流程,此处是模仿Redisson
的看门狗机制
。
代码实现:
// ...省略循环等代码
isSuccessLock = redisTemplate.opsForValue().setIfAbsent(redisKey, Thread.currentThread().getName(), lockTime, TimeUnit.MILLISECONDS);
if (Boolean.TRUE.equals(isSuccessLock)) {// 新增的重置过期时间方法 在加锁成功后, 开启1个线程, 做redis看门狗机制renewExpiration(lockKey, Thread.currentThread().getName(), lockTime);return true;
}
// 省略for循环中的try...catch代码
renewExpiration
刷新锁时间的方法实现
/*** lua脚本 更新锁时间, 如果已经获取不到值, 则不更新锁的过期时间*/
private static final String REDIS_RENEW_LOCK_SCRIPT = "if redis.call('get', KEYS[1]) == ARGV[1] " +"then return redis.call('expire', KEYS[1], ARGV[2]) " +"else return 0 end";
/*** 刷新锁时间* @param lockKey 锁key* @param requestId 锁线程号,redis的value* @param lockTime 锁时长*/
public void renewExpiration(String lockKey, String requestId, long lockTime) {DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>(REDIS_RENEW_LOCK_SCRIPT, Long.class);// 通过线程池创建异步线程taskExecutor.execute(() -> {while (true) {try {// 等待锁时长的1/3后刷新锁的过期时间Thread.sleep(lockTime / 3 * 1000);} catch (InterruptedException e) {Thread.currentThread().interrupt();break;}// 当前template只能传字符串, 将时间转为字符串传入String timeStr = String.valueOf(lockTime);// 延长过期时间原子操作Long execute = redisTemplate.execute(redisScript, new ArrayList<>(Collections.singleton(lockKey)), requestId, timeStr);if (execute == null || execute == 0) {break;}}});
}
总结:
其实后面才了解到Redis
的分布式锁直接引入Redisson
就万事大吉了,建议大家直接通过Redisson
去做Redis
的分布式锁,省时省力,方便完善。此处自己写其实也是为了更深入了解分布式锁的实现,而且一开始觉得就一个类解决的事情,引入多余的一个工具包会不会有点多余,结果做到后面才发现也有一些坑是自己没考虑周到的。等我后面有时间一定直接引入Redisson
。
参考链接:
redission的看门狗机制及应用