java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > ReentrantReadWriteLock不能锁升级

ReentrantReadWriteLock不能锁升级的原因总结

作者:skyline_wx

今天给大家带来的是关于Java并发的相关知识,文章围绕着为什么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

当这个tryAcquire返回false时,就跟ReentrantLock的逻辑差不多了,最后各种判断条件都会失败,最后,程序会阻塞在这里:java.util.concurrent.locks.AbstractQueuedSynchronizer#parkAndCheckInterrupt

parkAndCheckInterrupt

用流程图来描述一下这个问题是这样的:

在这里插入图片描述

假如只有一个线程t1,当t1已经获取读锁之后,再次获取写锁,因为写锁在加锁时判断到当前锁已经被加过读锁读写互斥,所以写锁会等待读锁释放之后再加锁。但是因为读锁是被当前线程持有的,所以这个等待会无限的等待下去,最后就成了死锁。

到此这篇关于ReentrantReadWriteLock不能锁升级的原因总结的文章就介绍到这了,更多相关ReentrantReadWriteLock不能锁升级内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文