解决 attempt to unlock lock, not locked by current thread by node id redisson异常

 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(); // 释放锁
                }
            }
        }

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值