线程并发redisson使用遇到的坑

Posted 技术能量站

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了线程并发redisson使用遇到的坑相关的知识,希望对你有一定的参考价值。

背景

因为业务上的一个购买需求,需要对库存进行行程保护,防止超卖的出现(我们不是电商公司),经过调研,最终选择使用Redission来进行控制。主要因为Redission丰富的API,开源框架,已经被广泛应用于实际生产环境。

问题描述

当我们使用Ression中Lock.lock()方法之后,如果存在线程并发常见情况下,会出现如下异常:


java.lang.IllegalMonitorStateException: attempt to unlock lock, not locked by current thread by node id: 9f178836-f7e1-44fe-a89d-2db52f399c0d thread-id: 22

问题分析

在thread-1还没有结束的时候,也就是在thread-1在获得锁但是还没有释放锁的时候, `thread-2由于被别的线程中断停止了等待从lock.tryLock的阻塞状态中返回继续执行接下来的逻辑,并且由于尝试去释放一个属于线程thread-1的锁而抛出了一个运行时异常导致该线程thread-2结束了, 然而thread-2完成了一系列操作后,线程thread-1才释放了自己的锁. 所以thread-2并没有获得锁,却执行了需要同步的内容,还尝试去释放锁。那解决方式我们就知道了,当前线程加的锁由当前线程去解锁,也就是说当我们使用lock.unlock的时候加上线程的判断即可。

问题解决

 public void unlock(String lockKey) 
    try 
      RLock lock = redissonClient.getLock(key);
      if(lock.isLocked()) // 是否还是锁定状态
           if(lock.isHeldByCurrentThread()) // 时候是当前执行线程的锁
        lock.unlock(); // 释放锁
         
       catch (Throwable e) 
           String msg = String.format("UNLOCK FAILED: key=%s", lockKey);
           throw new IllegalStateException(msg, e);
    

以上是关于线程并发redisson使用遇到的坑的主要内容,如果未能解决你的问题,请参考以下文章

并发编程精华问答| Java线程池使用时注意事项

并发编程精华问答| Java线程池使用时注意事项

这些并发容器的坑,你要谨记!

高并发你知道吗?大家都在使用Redisson实现分布式锁了!!

redis并发读写锁,使用Redisson实现分布式锁

RxJava使用过程中遇到线程相关的坑