本篇内容介绍了“java的ReentrantReadWriteLock不能锁升级的原因是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!
在ReentrantReadWriteLock
中,锁是不可以升级的,只能降级。
也就是如果当前线程持有了ReadLock
,那么就不能再获取WriteLock
,但是,如果当前线程持有了WriteLock
,可以直接获取ReadLock
下面用代码尝试一下:
Logger logger = LoggerFactory.getLogger(this.getClass()); ReentrantReadWriteLock reentrantReadWriteLock = new ReentrantReadWriteLock(); ReentrantReadWriteLock.ReadLock readLock = reentrantReadWriteLock.readLock(); ReentrantReadWriteLock.WriteLock writeLock = reentrantReadWriteLock.writeLock(); logger.info("线程:[{}],开始readLock",Thread.currentThread().getName()); readLock.lock(); logger.info("线程:[{}],readLock成功",Thread.currentThread().getName()); logger.info("线程:[{}],开始writeLock",Thread.currentThread().getName()); writeLock.lock(); logger.info("线程:[{}],writeLock成功",Thread.currentThread().getName());
从打印结果可以看出来,程序阻塞在了writeLock.lock();
这一行上。
下面我们看一下WriteLock的加锁过程的部分源码:
java.util.concurrent.locks.ReentrantReadWriteLock.Sync#tryAcquire
当这个tryAcquire返回false时,就跟ReentrantLock的逻辑差不多了,最后各种判断条件都会失败,最后,程序会阻塞在这里:java.util.concurrent.locks.AbstractQueuedSynchronizer#parkAndCheckInterrupt
用流程图来描述一下这个问题是这样的:
假如只有一个线程t1,当t1已经获取读锁
之后,再次获取写锁
,因为写锁
在加锁时判断到当前锁已经被加过读锁,读写互斥
,所以写锁
会等待读锁
释放之后再加锁。但是因为读锁
是被当前线程持有的,所以这个等待会无限的等待下去,最后就成了死锁。
“java的ReentrantReadWriteLock不能锁升级的原因是什么”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注亿速云网站,小编将为大家输出更多高质量的实用文章!
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。