Java读写锁,也就是ReentrantReadWriteLock
,其包含了读锁和写锁,其中读锁是可以多线程共享的,即共享锁,而写锁是排他锁,在更改时候不允许其他线程操作。读写锁底层是同一把锁(基于同一个AQS),所以会有同一时刻不允许读写锁共存的限制。
读写锁 示例代码如下:
public static void main(String[] args) {
ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
ReentrantReadWriteLock.WriteLock writeLock = lock.writeLock();
ReentrantReadWriteLock.ReadLock readLock = lock.readLock();
Thread t1 = new Thread(() -> {
readLock.lock();
System.out.println(Thread.currentThread().getName() + " read lock ok");
LockSupport.parkNanos(TimeUnit.SECONDS.toNanos(1));
readLock.unlock();
});
Thread t2 = new Thread(() -> {
readLock.lock();
System.out.println(Thread.currentThread().getName() + " read lock ok");
LockSupport.parkNanos(TimeUnit.SECONDS.toNanos(1));
readLock.unlock();
});
Thread t3 = new Thread(() -> {
writeLock.lock();
System.out.println(Thread.currentThread().getName() + " write lock ok");
writeLock.unlock();
});
t1.start();
t2.start();
t3.start();
}
输出结果为:
Thread-0 read lock ok
Thread-1 read lock ok
Thread-2 write lock ok // 1s后才打印
Java读写锁主要是基于AQS(队列同步器)的独占和共享来完成功能的,AQS使用一个int成员变量(private volatile int state)表示同步状态,通过内置的FIFO队列来完成资源获取线程的排队工作,并发包的作者(Doug Lea)期望它能够成为实现大部分同步需求的基础。关于AQS具体可以参考:AQS是如何控制线程的。
AQS只有一个int类型的state同步状态, 那它是如何维护独占和共享模式对应的