Java 并发之锁
Java 并发之锁
本文先阐述 Java 中各种锁的概念。
然后,重点介绍 Lock 和 Condition 两个接口及其实现。并发编程有两个核心问题:同步和互斥。
互斥,即同一时刻只允许一个线程访问共享资源;
同步,即线程之间如何通信、协作。
这两大问题,管程(
sychronized)都是能够解决的。J.U.C 包还提供了 Lock 和 Condition 两个接口来实现管程,其中 Lock 用于解决互斥问题,Condition 用于解决同步问题。
并发锁简介
确保线程安全最常见的做法是利用锁机制(Lock、sychronized)来对共享数据做互斥同步,这样在同一个时刻,只有一个线程可以执行某个方法或者某个代码块,那么操作必然是原子性的,线程安全的。
在工作、面试中,经常会听到各种五花八门的锁,听的人云里雾里。锁的概念术语很多,它们是针对不同的问题所提出的,通过简单的梳理,也不难理解。
可重入锁
可重入锁,顾名思义,指的是线程可以重复获取同一把锁。即同一个线程在外层方法获取了锁,在进入内层方法会自动获取锁。
可重入锁可以在一定程度上避免死锁。
ReentrantLock、ReentrantReadWriteLock是可重入锁。这点,从其命名也不难看出。synchronized也是一个可重入锁。
【示例】synchronized 的可重入示例
1 | synchronized void setA() throws Exception{ |
上面的代码就是一个典型场景:如果使用的锁不是可重入锁的话,setB 可能不会被当前线程执行,从而造成死锁。
【示例】ReentrantLock 的可重入示例
1 | class Task { |
公平锁与非公平锁
- 公平锁 - 公平锁是指 多线程按照申请锁的顺序来获取锁。
- 非公平锁 - 非公平锁是指 多线程不按照申请锁的顺序来获取锁 。这就可能会出现优先级反转(后来者居上)或者饥饿现象(某线程总是抢不过别的线程,导致始终无法执行)。
公平锁为了保证线程申请顺序,势必要付出一定的性能代价,因此其吞吐量一般低于非公平锁。
公平锁与非公平锁 在 Java 中的典型实现:
synchronized只支持非公平锁。ReentrantLock、ReentrantReadWriteLock,默认是非公平锁,但支持公平锁。
独占锁与共享锁
独占锁与共享锁是一种广义上的说法,从实际用途上来看,也常被称为互斥锁与读写锁。
- 独占锁 - 独占锁是指 锁一次只能被一个线程所持有。
- 共享锁 - 共享锁是指 锁可被多个线程所持有。
独占锁与共享锁在 Java 中的典型实现:
synchronized、ReentrantLock只支持独占锁。ReentrantReadWriteLock其写锁是独占锁,其读锁是共享锁。读锁是共享锁使得并发读是非常高效的,读写,写读 ,写写的过程是互斥的。
悲观锁与乐观锁
乐观锁与悲观锁不是指具体的什么类型的锁,而是处理并发同步的策略。
悲观锁(Pessimistic Lock)
- 总是假设最坏的情况,认为:不加锁的并发操作一定会出问题。
- 悲观锁在 Java 中的应用就是通过使用
synchronized和Lock显示加锁来进行互斥同步,这是一种阻塞同步。 - 悲观锁适合写操作频繁的场景。高并发的场景下,激烈的锁竞争会造成线程阻塞,大量阻塞线程会导致系统的上下文切换,增加系统的性能开销。并且,悲观锁还可能会存在死锁问题,影响代码的正常运行。
【示例】悲观锁示例
1 | public void syncTask() { |
乐观锁(OptimisticLock)
- 乐观锁总是假设最好的情况,认为:不加锁的并发操作也没什么问题。每次访问数据时,都假设数据不会被其他线程修改,不必加锁。虽然不加锁,但不意味着什么都不做,而是在更新的时候,判断一下在此期间是否有其他线程更新该数据。
- 乐观锁最常见的实现方式,是使用版本号机制或 CAS 算法(Compare And Swap)去实现。Java 中的原子类就是基于 CAS 实现的。
- 乐观锁的优点是:减少锁竞争,提高并发度。
- 乐观锁的缺点是:
- 存在 ABA 问题。所谓的 ABA 问题是指在并发编程中,如果一个变量初次读取的时候是 A 值,它的值被改成了 B,然后又其他线程把 B 值改成了 A,而另一个早期线程在对比值时会误以为此值没有发生改变,但其实已经发生变化了
- 如果乐观锁所检查的数据存在大量锁竞争,会由于不断循环重试,产生大量的 CPU 开销。
- 乐观锁适合读多写少的场景。高并发的场景下,乐观锁相比悲观锁来说,不存在锁竞争造成线程阻塞,也不会有死锁的问题,在性能上往往会更胜一筹。但是,如果冲突频繁发生(写占比非常多的情况),会频繁失败和重试,这样同样会非常影响性能,导致 CPU 飙升。
【示例】乐观锁示例
1 | // AtomicInteger 的 getAndAccumulate 方法采用了自旋 + CAS 的乐观锁模式 |
乐观锁也是一种通用的锁机制,不仅在 Java 中,在其他很多软件领域,也存在乐观锁机制。比如下面的示例是 MySQL 中的乐观锁示例。
假设,order 表中有一个字段 status,表示订单状态:status 为 1 代表订单未支付;status 为 2 代表订单已支付。现在,要将 id 为 1 的订单状态置为已支付,则操作如下:
1 | select status, version from order where id=#{id} |
偏向锁、轻量级锁、重量级锁
所谓轻量级锁与重量级锁,指的是锁控制粒度的粗细。显然,控制粒度越细,阻塞开销越小,并发性也就越高。
Java 1.6 以前,重量级锁一般指的是 synchronized ,而轻量级锁指的是 volatile。
Java 1.6 以后,针对 synchronized 做了大量优化,引入 4 种锁状态: 无锁状态、偏向锁、轻量级锁和重量级锁。锁可以单向的从偏向锁升级到轻量级锁,再从轻量级锁升级到重量级锁 。
- 偏向锁 - 偏向锁是指一段同步代码一直被一个线程所访问,那么该线程会自动获取锁。降低获取锁的代价。
- 轻量级锁 - 是指当锁是偏向锁的时候,被另一个线程所访问,偏向锁就会升级为轻量级锁,其他线程会通过自旋的形式尝试获取锁,不会阻塞,提高性能。
- 重量级锁 - 是指当锁为轻量级锁的时候,另一个线程虽然是自旋,但自旋不会一直持续下去,当自旋一定次数的时候,还没有获取到锁,就会进入阻塞,该锁膨胀为重量级锁。重量级锁会让其他申请的线程进入阻塞,性能降低。
分段锁
分段锁其实是一种锁的设计,并不是具体的一种锁。所谓分段锁,就是把锁的对象分成多段,每段独立控制,使得锁粒度更细,减少阻塞开销,从而提高并发性。这其实很好理解,就像高速公路上的收费站,如果只有一个收费口,那所有的车只能排成一条队缴费;如果有多个收费口,就可以分流了。
Hashtable 使用 synchronized 修饰方法来保证线程安全性,那么面对线程的访问,Hashtable 就会锁住整个对象,所有的其它线程只能等待,这种阻塞方式的吞吐量显然很低。
Java 1.7 以前的 ConcurrentHashMap 就是分段锁的典型案例。ConcurrentHashMap 维护了一个 Segment 数组,一般称为分段桶。
1 | final Segment<K,V>[] segments; |
当有线程访问 ConcurrentHashMap 的数据时,ConcurrentHashMap 会先根据 hashCode 计算出数据在哪个桶(即哪个 Segment),然后锁住这个 Segment。
内置锁和显示锁
Java 1.5 之前,协调对共享对象的访问时可以使用的机制只有 synchronized 和 volatile。这两个都属于内置锁,即锁的申请和释放都是由 JVM 所控制。
Java 1.5 之后,增加了新的机制:ReentrantLock、ReentrantReadWriteLock ,这类锁的申请和释放都可以由程序所控制,所以常被称为显示锁。
💡
synchronized的用法和原理可以参考:Java 并发之内存模型 。:bell: 注意:如果不需要
ReentrantLock、ReentrantReadWriteLock所提供的高级同步特性,**应该优先考虑使用synchronized**。理由如下:
- Java 1.6 以后,
synchronized做了大量的优化,其性能已经与ReentrantLock、ReentrantReadWriteLock基本上持平。- 从趋势来看,Java 未来更可能会优化
synchronized,而不是ReentrantLock、ReentrantReadWriteLock,因为synchronized是 JVM 内置属性,它能执行一些优化。ReentrantLock、ReentrantReadWriteLock申请和释放锁都是由程序控制,如果使用不当,可能造成死锁,这是很危险的。
以下对比一下显示锁和内置锁的差异:
- 主动获取锁和释放锁
synchronized不能主动获取锁和释放锁。获取锁和释放锁都是 JVM 控制的。ReentrantLock可以主动获取锁和释放锁。(如果忘记释放锁,就可能产生死锁)。
- 响应中断
synchronized不能响应中断。ReentrantLock可以响应中断。
- 超时机制
synchronized没有超时机制。ReentrantLock有超时机制。ReentrantLock可以设置超时时间,超时后自动释放锁,避免一直等待。
- 支持公平锁
synchronized只支持非公平锁。ReentrantLock支持非公平锁和公平锁。
- 是否支持共享
- 被
synchronized修饰的方法或代码块,只能被一个线程访问(独享)。如果这个线程被阻塞,其他线程也只能等待 ReentrantLock可以基于Condition灵活的控制同步条件。
- 被
- 是否支持读写分离
synchronized不支持读写锁分离;ReentrantReadWriteLock支持读写锁,从而使阻塞读写的操作分开,有效提高并发性。
Lock 和 Condition
为何引入 Lock 和 Condition
并发编程领域,有两大核心问题:一个是互斥,即同一时刻只允许一个线程访问共享资源;另一个是同步,即线程之间如何通信、协作。这两大问题,管程都是能够解决的。Java SDK 并发包通过 Lock 和 Condition 两个接口来实现管程,其中 Lock 用于解决互斥问题,Condition 用于解决同步问题。
synchronized 是管程的一种实现,既然如此,何必再提供 Lock 和 Condition。
JDK 1.6 以前,synchronized 还没有做优化,性能远低于 Lock。但是,性能不是引入 Lock 的最重要因素。真正关键在于:synchronized 使用不当,可能会出现死锁。synchronized 无法通过破坏不可抢占条件来避免死锁。原因是 synchronized 申请资源的时候,如果申请不到,线程直接进入阻塞状态了,而线程进入阻塞状态,啥都干不了,也释放不了线程已经占有的资源。
与内置锁 synchronized 不同的是,**Lock 提供了一组无条件的、可轮询的、定时的以及可中断的锁操作**,所有获取锁、释放锁的操作都是显式的操作。
- 能够响应中断。synchronized 的问题是,持有锁 A 后,如果尝试获取锁 B 失败,那么线程就进入阻塞状态,一旦发生死锁,就没有任何机会来唤醒阻塞的线程。但如果阻塞状态的线程能够响应中断信号,也就是说当我们给阻塞的线程发送中断信号的时候,能够唤醒它,那它就有机会释放曾经持有的锁 A。这样就破坏了不可抢占条件了。
- 支持超时。如果线程在一段时间之内没有获取到锁,不是进入阻塞状态,而是返回一个错误,那这个线程也有机会释放曾经持有的锁。这样也能破坏不可抢占条件。
- 非阻塞地获取锁。如果尝试获取锁失败,并不进入阻塞状态,而是直接返回,那这个线程也有机会释放曾经持有的锁。这样也能破坏不可抢占条件。
Lock 接口
Lock 的接口定义如下:
1 | public interface Lock { |
lock()- 获取锁。unlock()- 释放锁。tryLock()- 尝试获取锁,仅在调用时锁未被另一个线程持有的情况下,才获取该锁。tryLock(long time, TimeUnit unit)- 和tryLock()类似,区别仅在于限定时间,如果限定时间内未获取到锁,视为失败。lockInterruptibly()- 锁未被另一个线程持有,且线程没有被中断的情况下,才能获取锁。newCondition()- 返回一个绑定到Lock对象上的Condition实例。
Condition
Condition 实现了管程模型里面的条件变量。
前文中提过 Lock 接口中 有一个 newCondition() 方法用于返回一个绑定到 Lock 对象上的 Condition 实例。Condition 是什么?有什么作用?本节将一一讲解。
在单线程中,一段代码的执行可能依赖于某个状态,如果不满足状态条件,代码就不会被执行(典型的场景,如:if ... else ...)。在并发环境中,当一个线程判断某个状态条件时,其状态可能是由于其他线程的操作而改变,这时就需要有一定的协调机制来确保在同一时刻,数据只能被一个线程锁修改,且修改的数据状态被所有线程所感知。
Java 1.5 之前,主要是利用 Object 类中的 wait、notify、notifyAll 配合 synchronized 来进行线程间通信。wait、notify、notifyAll 需要配合 synchronized 使用,不适用于 Lock。而使用 Lock 的线程,彼此间通信应该使用 Condition 。这可以理解为,什么样的锁配什么样的钥匙。内置锁(synchronized)配合内置条件队列(wait、notify、notifyAll ),显式锁(Lock)配合显式条件队列(Condition )。
Condition 的特性
Condition 接口定义如下:
1 | public interface Condition { |
其中,await、signal、signalAll 与 wait、notify、notifyAll 相对应,功能也相似。除此以外,Condition 相比内置条件队列( wait、notify、notifyAll ),提供了更为丰富的功能:
- 每个锁(
Lock)上可以存在多个Condition,这意味着锁的状态条件可以有多个。 - 支持公平的或非公平的队列操作。
- 支持可中断的条件等待,相关方法:
awaitUninterruptibly()。 - 支持可定时的等待,相关方法:
awaitNanos(long)、await(long, TimeUnit)、awaitUntil(Date)。
Condition 的用法
这里以 Condition 来实现一个消费者、生产者模式。
:bell: 注意:事实上,解决此类问题使用
CountDownLatch、Semaphore等工具更为便捷、安全。想了解详情,可以参考 Java 并发工具类 。
产品类
1 | class Message { |
消费者
1 | class MessageConsumer implements Runnable { |
生产者
1 | class MessageProducer implements Runnable { |
测试
1 | public class LockConditionDemo { |
ReentrantLock
ReentrantLock 类是 Lock 接口的具体实现,与内置锁 synchronized 相同的是,它是一个可重入锁。
ReentrantLock 的特性如下:
ReentrantLock提供了与synchronized相同的互斥性、内存可见性和可重入性。ReentrantLock支持公平锁和非公平锁(默认)两种模式。ReentrantLock实现了Lock接口,支持了synchronized所不具备的灵活性,增加了轮询、超时、中断等功能。synchronized无法中断一个正在等待获取锁的线程synchronized无法在请求获取一个锁时无休止地等待
ReentrantLock 的用法
前文了解了 ReentrantLock 的特性,接下来,我们要讲述其具体用法。
ReentrantLock 的构造方法
ReentrantLock 有两个构造方法:
1 | public ReentrantLock() {} |
ReentrantLock()- 默认构造方法会初始化一个非公平锁(NonfairSync);ReentrantLock(boolean)-new ReentrantLock(true)会初始化一个公平锁(FairSync)。
lock 和 unlock 方法
lock()- 无条件获取锁。如果当前线程无法获取锁,则当前线程进入休眠状态不可用,直至当前线程获取到锁。如果该锁没有被另一个线程持有,则获取该锁并立即返回,将锁的持有计数设置为 1。unlock()- 用于释放锁。
:bell: 注意:请务必牢记,获取锁操作
lock()必须在try catch块中进行,并且将释放锁操作unlock()放在finally块中进行,以保证锁一定被被释放,防止死锁的发生。
示例:ReentrantLock 的基本操作
1 | public class ReentrantLockDemo { |
输出结果:
1 | java.util.concurrent.locks.ReentrantLock@64fcd88a[Locked by thread Thread-A] |
tryLock 方法
与无条件获取锁相比,tryLock 有更完善的容错机制。
tryLock()- 可轮询获取锁。如果成功,则返回 true;如果失败,则返回 false。也就是说,这个方法无论成败都会立即返回,获取不到锁(锁已被其他线程获取)时不会一直等待。tryLock(long, TimeUnit)- 可定时获取锁。和tryLock()类似,区别仅在于这个方法在获取不到锁时会等待一定的时间,在时间期限之内如果还获取不到锁,就返回 false。如果如果一开始拿到锁或者在等待期间内拿到了锁,则返回 true。
示例:ReentrantLock 的 tryLock() 操作
修改上个示例中的 execute() 方法
1 | public void execute() { |
示例:ReentrantLock 的 tryLock(long, TimeUnit) 操作
修改上个示例中的 execute() 方法
1 | public void execute() { |
lockInterruptibly 方法
lockInterruptibly()- 可中断获取锁。可中断获取锁可以在获得锁的同时保持对中断的响应。可中断获取锁比其它获取锁的方式稍微复杂一些,需要两个try-catch块(如果在获取锁的操作中抛出了InterruptedException,那么可以使用标准的try-finally加锁模式)。- 举例来说:假设有两个线程同时通过
lock.lockInterruptibly()获取某个锁时,若线程 A 获取到了锁,则线程 B 只能等待。若此时对线程 B 调用threadB.interrupt()方法能够中断线程 B 的等待过程。由于lockInterruptibly()的声明中抛出了异常,所以lock.lockInterruptibly()必须放在try块中或者在调用lockInterruptibly()的方法外声明抛出InterruptedException。
- 举例来说:假设有两个线程同时通过
:bell: 注意:当一个线程获取了锁之后,是不会被
interrupt()方法中断的。单独调用interrupt()方法不能中断正在运行状态中的线程,只能中断阻塞状态中的线程。因此当通过lockInterruptibly()方法获取某个锁时,如果未获取到锁,只有在等待的状态下,才可以响应中断。
示例:ReentrantLock 的 lockInterruptibly() 操作
修改上个示例中的 execute() 方法
1 | public void execute() { |
newCondition 方法
newCondition() - 返回一个绑定到 Lock 对象上的 Condition 实例。Condition 的特性和具体方法请阅读下文 [Condition](#五 condition)。
ReentrantLock 的原理
ReentrantLock 的可见性
1 | class X { |
ReentrantLock,内部持有一个 volatile 的成员变量 state,获取锁的时候,会读写 state 的值;解锁的时候,也会读写 state 的值(简化后的代码如下面所示)。也就是说,在执行 value+=1 之前,程序先读写了一次 volatile 变量 state,在执行 value+=1 之后,又读写了一次 volatile 变量 state。根据相关的 Happens-Before 规则:
- 顺序性规则:对于线程 T1,value+=1 Happens-Before 释放锁的操作 unlock();
- volatile 变量规则:由于 state = 1 会先读取 state,所以线程 T1 的 unlock() 操作 Happens-Before 线程 T2 的 lock() 操作;
- 传递性规则:线程 T1 的 value+=1 Happens-Before 线程 T2 的 lock() 操作。
ReentrantLock 的数据结构
阅读 ReentrantLock 的源码,可以发现它有一个核心字段:
1 | private final Sync sync; |
sync- 内部抽象类ReentrantLock.Sync对象,Sync继承自 AQS。它有两个子类:ReentrantLock.FairSync- 公平锁。ReentrantLock.NonfairSync- 非公平锁。
查看源码可以发现,ReentrantLock 实现 Lock 接口其实是调用 ReentrantLock.FairSync 或 ReentrantLock.NonfairSync 中各自的实现,这里不一一列举。
ReentrantLock 的获取锁和释放锁
ReentrantLock 获取锁和释放锁的接口,从表象看,是调用 ReentrantLock.FairSync 或 ReentrantLock.NonfairSync 中各自的实现;从本质上看,是基于 AQS 的实现。
仔细阅读源码很容易发现:
void lock()调用 Sync 的 lock() 方法。void lockInterruptibly()直接调用 AQS 的 [获取可中断的独占锁](#获取可中断的独占锁) 方法lockInterruptibly()。boolean tryLock()调用 Sync 的nonfairTryAcquire()。boolean tryLock(long time, TimeUnit unit)直接调用 AQS 的 [获取超时等待式的独占锁](#获取超时等待式的独占锁) 方法tryAcquireNanos(int arg, long nanosTimeout)。void unlock()直接调用 AQS 的 [释放独占锁](#释放独占锁) 方法release(int arg)。
直接调用 AQS 接口的方法就不再赘述了,其原理在 [AQS 的原理](#AQS 的原理) 中已经用很大篇幅进行过讲解。
nonfairTryAcquire 方法源码如下:
1 | // 公平锁和非公平锁都会用这个方法区尝试获取锁 |
处理流程很简单:
- 如果同步状态为 0,设置同步状态设为 acquires,并设置当前线程为排它线程,然后返回 true,获取锁成功。
- 如果同步状态不为 0 且当前线程为排它线程,设置同步状态为当前状态值+acquires 值,然后返回 true,获取锁成功。
- 否则,返回 false,获取锁失败。
公平锁和非公平锁
ReentrantLock 这个类有两个构造函数,一个是无参构造函数,一个是传入 fair 参数的构造函数。fair 参数代表的是锁的公平策略,如果传入 true 就表示需要构造一个公平锁,反之则表示要构造一个非公平锁。
锁都对应着一个等待队列,如果一个线程没有获得锁,就会进入等待队列,当有线程释放锁的时候,就需要从等待队列中唤醒一个等待的线程。如果是公平锁,唤醒的策略就是谁等待的时间长,就唤醒谁,很公平;如果是非公平锁,则不提供这个公平保证,有可能等待时间短的线程反而先被唤醒。
lock 方法在公平锁和非公平锁中的实现:
二者的区别仅在于申请非公平锁时,如果同步状态为 0,尝试将其设为 1,如果成功,直接将当前线程置为排它线程;否则和公平锁一样,调用 AQS 获取独占锁方法 acquire。
1 | // 非公平锁实现 |
ReentrantReadWriteLock
ReadWriteLock 适用于读多写少的场景。
ReentrantReadWriteLock 类是 ReadWriteLock 接口的具体实现,它是一个可重入的读写锁。ReentrantReadWriteLock 维护了一对读写锁,将读写锁分开,有利于提高并发效率。
读写锁,并不是 Java 语言特有的,而是一个广为使用的通用技术,所有的读写锁都遵守以下三条基本原则:
- 允许多个线程同时读共享变量;
- 只允许一个线程写共享变量;
- 如果一个写线程正在执行写操作,此时禁止读线程读共享变量。
读写锁与互斥锁的一个重要区别就是读写锁允许多个线程同时读共享变量,而互斥锁是不允许的,这是读写锁在读多写少场景下性能优于互斥锁的关键。但读写锁的写操作是互斥的,当一个线程在写共享变量的时候,是不允许其他线程执行写操作和读操作。
ReentrantReadWriteLock 的特性
ReentrantReadWriteLock 的特性如下:
ReentrantReadWriteLock适用于读多写少的场景。如果是写多读少的场景,由于ReentrantReadWriteLock其内部实现比ReentrantLock复杂,性能可能反而要差一些。如果存在这样的问题,需要具体问题具体分析。由于ReentrantReadWriteLock的读写锁(ReadLock、WriteLock)都实现了Lock接口,所以要替换为ReentrantLock也较为容易。ReentrantReadWriteLock实现了ReadWriteLock接口,支持了ReentrantLock所不具备的读写锁分离。ReentrantReadWriteLock维护了一对读写锁(ReadLock、WriteLock)。将读写锁分开,有利于提高并发效率。ReentrantReadWriteLock的加锁策略是:允许多个读操作并发执行,但每次只允许一个写操作。ReentrantReadWriteLock为读写锁都提供了可重入的加锁语义。ReentrantReadWriteLock支持公平锁和非公平锁(默认)两种模式。
ReadWriteLock 接口定义如下:
1 | public interface ReadWriteLock { |
readLock- 返回用于读操作的锁(ReadLock)。writeLock- 返回用于写操作的锁(WriteLock)。
在读写锁和写入锁之间的交互可以采用多种实现方式,ReadWriteLock 的一些可选实现包括:
- 释放优先 - 当一个写入操作释放写锁,并且队列中同时存在读线程和写线程,那么应该优先选择读线程、写线程,还是最先发出请求的线程?
- 读线程插队 - 如果锁是由读线程持有,但有写线程正在等待,那么新到达的读线程能否立即获得访问权,还是应该在写线程后面等待?如果允许读线程插队到写线程之前,那么将提高并发性,但可能造成线程饥饿问题。
- 重入性 - 读锁和写锁是否是可重入的?
- 降级 - 如果一个线程持有写入锁,那么它能否在不释放该锁的情况下获得读锁?这可能会使得写锁被降级为读锁,同时不允许其他写线程修改被保护的资源。
- 升级 - 读锁能否优先于其他正在等待的读线程和写线程而升级为一个写锁?在大多数的读写锁实现中并不支持升级,因为如果没有显式的升级操作,那么很容易造成死锁。
ReentrantReadWriteLock 的用法
前文了解了 ReentrantReadWriteLock 的特性,接下来,我们要讲述其具体用法。
ReentrantReadWriteLock 的构造方法
ReentrantReadWriteLock 和 ReentrantLock 一样,也有两个构造方法,且用法相似。
1 | public ReentrantReadWriteLock() {} |
ReentrantReadWriteLock()- 默认构造方法会初始化一个非公平锁(NonfairSync)。在非公平的锁中,线程获得锁的顺序是不确定的。写线程降级为读线程是可以的,但读线程升级为写线程是不可以的(这样会导致死锁)。ReentrantReadWriteLock(boolean)-new ReentrantLock(true)会初始化一个公平锁(FairSync)。对于公平锁,等待时间最长的线程将优先获得锁。如果这个锁是读线程持有,则另一个线程请求写锁,那么其他读线程都不能获得读锁,直到写线程释放写锁。
ReentrantReadWriteLock 的使用实例
在 [ReentrantReadWriteLock 的特性](#reentrantreadwritelock-的特性) 中已经介绍过,ReentrantReadWriteLock 的读写锁(ReadLock、WriteLock) 都实现了 Lock 接口,所以其各自独立的使用方式与 ReentrantLock 一样,这里不再赘述。
ReentrantReadWriteLock 与 ReentrantLock 用法上的差异,主要在于读写锁的配合使用。本文以一个典型使用场景来进行讲解。
【示例】基于 ReadWriteLock 实现一个简单的泛型无界缓存
1 | /** |
说明:
- 使用
WeakHashMap而不是HashMap来存储键值对。WeakHashMap中存储的对象是弱引用,JVM GC 时会自动清除没有被引用的弱引用对象。 - 向
Map写数据前加写锁,写完后,释放写锁。 - 向
Map读数据前加读锁,读完后,释放读锁。
测试其线程安全性:
1 | /** |
说明:示例中,通过线程池启动 20 个并发任务。任务每次向缓存中写入 3 个随机值,key 固定;然后主线程每次固定读取缓存中第一个 key 的值。
输出结果:
1 | main 读数据 0:null |
ReentrantReadWriteLock 的原理
前面了解了 ReentrantLock 的原理,理解 ReentrantReadWriteLock 就容易多了。
ReentrantReadWriteLock 的数据结构
阅读 ReentrantReadWriteLock 的源码,可以发现它有三个核心字段:
1 | /** Inner class providing readlock */ |
sync- 内部类ReentrantReadWriteLock.Sync对象。与ReentrantLock类似,它有两个子类:ReentrantReadWriteLock.FairSync和ReentrantReadWriteLock.NonfairSync,分别表示公平锁和非公平锁的实现。readerLock- 内部类ReentrantReadWriteLock.ReadLock对象,这是一把读锁。writerLock- 内部类ReentrantReadWriteLock.WriteLock对象,这是一把写锁。
ReentrantReadWriteLock 的获取锁和释放锁
1 | public static class ReadLock implements Lock, java.io.Serializable { |
StampedLock
ReadWriteLock 支持两种模式:一种是读锁,一种是写锁。而 StampedLock 支持三种模式,分别是:写锁、悲观读锁和乐观读。其中,写锁、悲观读锁的语义和 ReadWriteLock 的写锁、读锁的语义非常类似,允许多个线程同时获取悲观读锁,但是只允许一个线程获取写锁,写锁和悲观读锁是互斥的。不同的是:StampedLock 里的写锁和悲观读锁加锁成功之后,都会返回一个 stamp;然后解锁的时候,需要传入这个 stamp。
注意这里,用的是“乐观读”这个词,而不是“乐观读锁”,是要提醒你,乐观读这个操作是无锁的,所以相比较 ReadWriteLock 的读锁,乐观读的性能更好一些。
StampedLock 的性能之所以比 ReadWriteLock 还要好,其关键是 StampedLock 支持乐观读的方式。
- ReadWriteLock 支持多个线程同时读,但是当多个线程同时读的时候,所有的写操作会被阻塞;
- 而 StampedLock 提供的乐观读,是允许一个线程获取写锁的,也就是说不是所有的写操作都被阻塞。
对于读多写少的场景 StampedLock 性能很好,简单的应用场景基本上可以替代 ReadWriteLock,但是,StampedLock 的功能仅仅是 ReadWriteLock 的子集,在使用的时候,还是有几个地方需要注意一下。
- StampedLock 不支持重入
- StampedLock 的悲观读锁、写锁都不支持条件变量。
- 如果线程阻塞在 StampedLock 的 readLock() 或者 writeLock() 上时,此时调用该阻塞线程的 interrupt() 方法,会导致 CPU 飙升。**使用 StampedLock 一定不要调用中断操作,如果需要支持中断功能,一定使用可中断的悲观读锁 readLockInterruptibly() 和写锁 writeLockInterruptibly()**。
【示例】StampedLock 阻塞时,调用 interrupt() 导致 CPU 飙升
1 | final StampedLock lock = new StampedLock(); |
【示例】StampedLock 读模板:
1 | final StampedLock sl = new StampedLock(); |
【示例】StampedLock 写模板:
1 | long stamp = sl.writeLock(); |