ReentrantReadWriteLock不能锁升级的原因总结
为什么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已经获取读锁
之后,再次获取写锁
,因为写锁
在加锁时判断到当前锁已经被加过读锁,读写互斥
,所以写锁
会等待读锁
释放之后再加锁。但是因为读锁
是被当前线程持有的,所以这个等待会无限的等待下去,最后就成了死锁。
到此这篇关于ReentrantReadWriteLock不能锁升级的原因总结的文章就介绍到这了,更多相关ReentrantReadWriteLock不能锁升级内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
Java中的ArrayList.trimToSize()方法详解
这篇文章主要介绍了Java中的ArrayList.trimToSize()方法详解,前几天看了Java ArrayList,没有明白trimToSize()这个方法是什么意思,所以看了一下源码并且debug一下自己的一个例子,明白了其中的含义,需要的朋友可以参考下2023-11-11SpringBoot @PostConstruct原理用法解析
这篇文章主要介绍了SpringBoot @PostConstruct原理用法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下2020-08-08SpringBoot与Spring中数据缓存Cache超详细讲解
我们知道内存读取速度远大于硬盘读取速度,当需要重复获取相同数据时,一次一次的请求数据库或者远程服务,导致在数据库查询或者远程方法调用上小号大量的时间,最终导致程序性能降低,这就是数据缓存要解决的问题,学过计算机组成原理或者操作系统的同学们应该比较熟悉2022-10-10
最新评论