Redission分布式锁进行unlock操作时,会提示这个异常,源码如下:

    @Override
    public void unlock() {
        Boolean opStatus = get(unlockInnerAsync(Thread.currentThread().getId()));
        if (opStatus == null) {
            throw new IllegalMonitorStateException("attempt to unlock lock, not locked by current thread by node id: "
                    + id + " thread-id: " + Thread.currentThread().getId());
        }
        
        ******
}

为什么会这样,由于在进行lock操作时,会设置一个时间默认30s,

1.当你在完成lock后,里面的业务代码执行时间大于lock时间时,进行unlock,会抛出该异常

2.多线程竞争的问题,当第一个线程完成lock,此时并未 unlock,如此,第二个线程尝试获取锁,并进行lock操作,会抛出该异常。

3.多线程竞争的问题,当第一个线程完成lock,此时并未 unlock,第二个线程未获取到锁,但执行finally,第二线程尝试着去释放第一个线程的锁。就是问题的翻译如下:尝试去释放不是当前线程持有的锁。

解决办法:

在lock或unlock前,判断下状态合法性即可,而非直接进行加锁解锁操作,可以参考如下(如有不对,欢迎指正):

            RLock lock = null;
        try {
            lock = redissonClient.getLock(key);
            if (lock.isLocked()) { // 是否还是锁定状态
                retrun;
            }
            //也可以指定锁时间的
            boolean b = lock.tryLock();
            ....
        } catch (Exception e) {

        } finally {
            if (null != lock && lock.isLocked()) { // 是否还是锁定状态
                if (lock.isHeldByCurrentThread()) {//是否是当前执行线程的锁
                    lock.unlock(); // 释放锁
                }
            }
        }

Logo

旨在为数千万中国开发者提供一个无缝且高效的云端环境,以支持学习、使用和贡献开源项目。

更多推荐