并发编程之深入理解AQS之独占锁ReentrantLock
一、AQS原理分析
1.1 什么是AQS
java.util.concurrent包中的大多数同步器实现都是围绕着共同的基础行为,比如等待队列、条件队列、独占获取、共享获取等,而这些行为的抽象就是基于AbstractQueuedSynchronizer(简称AQS)实现的,AQS是一个抽象同步框架,可以用来实现一个依赖状态的同步器。
JDK中提供的大多数的同步器如Lock, Latch, Barrier等,都是基于AQS框架来实现的
1.一般是通过一个内部类Sync继承 AQS
2.将同步器所有调用都映射到Sync对应的方法
AQS具备的特性:
(1).阻塞等待队列
(2).共享/独占
(3).公平/非公平
(4).可重入
(5).允许中断
AQS内部维护属性volatile int state
state表示资源的可用状态
State三种访问方式:
(1).getState()
(2).setState()
(3).compareAndSetState()
AQS定义两种资源共享方式
Exclusive-独占,只有一个线程能执行,如ReentrantLock
Share-共享,多个线程可以同时执行,如Semaphore/CountDownLatch
AQS定义两种队列
同步等待队列: 主要用于维护获取锁失败时入队的线程
条件等待队列: 调用await()的时候会释放锁,然后线程会加入到条件队列,调用signal()唤醒的时候会把条件队列中的线程节点移动到同步队列中,等待再次获得锁
AQS 定义了5个队列中节点状态:
(1).值为0,初始化状态,表示当前节点在sync队列中,等待着获取锁;
(2).CANCELLED,值为1,表示当前的线程被取消;
(3).SIGNAL,值为-1,表示当前节点的后继节点包含的线程需要运行,也就是unpark;
(4).CONDITION,值为-2,表示当前节点在等待condition,也就是在condition队列中;
(5).PROPAGATE,值为-3,表示当前场景下后续的acquireShared能够得以执行;
不同的自定义同步器竞争共享资源的方式也不同。自定义同步器在实现时只需要实现共享资源state的获取与释放方式即可,至于具体线程等待队列的维护(如获取资源失败入队/唤醒出队等),AQS已经在顶层实现好了。自定义同步器实现时主要实现以下几种方法:
(1).isHeldExclusively():该线程是否正在独占资源。只有用到condition才需要去实现它。
(2).tryAcquire(int):独占方式。尝试获取资源,成功则返回true,失败则返回false。
(3).tryRelease(int):独占方式。尝试释放资源,成功则返回true,失败则返回false。
(4).tryAcquireShared(int):共享方式。尝试获取资源。负数表示失败;0表示成功,但没有剩余可用资源;正数表示成功,且有剩余资源。
(5).tryReleaseShared(int):共享方式。尝试释放资源,如果释放后允许唤醒后续等待结点返回true,否则返回false。
1.2 同步等待队列
AQS当中的同步等待队列也称CLH队列,CLH队列是Craig、Landin、Hagersten三人发明的一种基于双向链表数据结构的队列,是FIFO先进先出线程等待队列,Java中的CLH队列是原CLH队列的一个变种,线程由原自旋机制改为阻塞机制。
AQS 依赖CLH同步队列来完成同步状态的管理:
(1).当前线程如果获取同步状态失败时,AQS则会将当前线程已经等待状态等信息构造成一个节点(Node)并将其加入到CLH同步队列,同时会阻塞当前线程;
(2).当同步状态释放时,会把首节点唤醒(公平锁),使其再次尝试获取同步状态;
(3).通过signal或signalAll将条件队列中的节点转移到同步队列。(由条件队列转化为同步队列)
1.3 条件等待队列
AQS中条件队列是使用单向列表保存的,用nextWaiter来连接:
(1).调用await方法阻塞线程;
(2).当前线程存在于同步队列的头结点,调用await方法进行阻塞(从同步队列转化到条件队列);
条件队列与同步队列总结:
1.4 Condition接口详解
调用Condition#await方法会释放当前持有的锁,然后阻塞当前线程,同时向Condition队列尾部添加一个节点,所以调用Condition#await方法的时候必须持有锁。
调用Condition#signal方法会将Condition队列的首节点移动到阻塞队列尾部,然后唤醒因调用Condition#await方法而阻塞的线程(唤醒之后这个线程就可以去竞争锁了),所以调用Condition#signal方法的时候必须持有锁,持有锁的线程唤醒被因调用Condition#await方法而阻塞的线程。
等待唤醒机制之await/signal测试
@Slf4j
public class ConditionTest {
public static void main(String[] args) {
Lock lock = new ReentrantLock();
Condition condition = lock.newCondition();
new Thread(() -> {
lock.lock();
try {
log.debug(Thread.currentThread().getName() + " 开始处理任务");
condition.await();
log.debug(Thread.currentThread().getName() + " 结束处理任务");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}).start();
new Thread(() -> {
lock.lock();
try {
log.debug(Thread.currentThread().getName() + " 开始处理任务");
Thread.sleep(2000);
condition.signal();
log.debug(Thread.currentThread().getName() + " 结束处理任务");
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}).start();
}
二、ReentrantLock详解
ReentrantLock是一种基于AQS框架的应用实现,是JDK中的一种线程并发访问的同步手段,它的功能类似于synchronized是一种互斥锁,可以保证线程安全。
相对于 synchronized, ReentrantLock具备如下特点:
(1).可中断
(2).可以设置超时时间
(3).可以设置为公平锁
(4).支持多个条件变量
(5).与 synchronized 一样,都支持可重入
2.1 ReentrantLock的使用
2.1.1 同步执行(类似于synchronized)
public class ReentrantLockDemo {
private static int sum = 0;
//private static TulingLock lock = new TulingLock();
private static Lock lock = new ReentrantLock();
public static void main(String[] args) {
for (int i = 0; i < 3; i++) {
Thread thread = new Thread(()->{
//加锁
lock.lock();
try {
// 临界区代码
// TODO 业务逻辑:读写操作不能保证线程安全
for (int j = 0; j < 10000; j++) {
sum++;
}
}finally {
// 解锁
lock.unlock();
}
});
thread.start();
}
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("sum="+sum);
}
}
运行结果:
sum=30000
2.1.2 可重入
@Slf4j
public class ReentrantLockDemo2 {
private static ReentrantLock lock = new ReentrantLock();
public static void main(String[] args) {
method1();
}
public static void method1(){
lock.lock();
try {
log.debug("execute method1");
method2();
}finally {
lock.unlock();
}
}
public static void method2(){
lock.lock();
try {
log.debug("execute method2");
method3();
}finally {
lock.unlock();
}
}
public static void method3(){
lock.lock();
try {
log.debug("execute method3");
}finally {
lock.unlock();
}
}
}
运行结果:
[main] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo2 - execute method1
[main] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo2 - execute method2
[main] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo2 - execute method3
2.1.3 可中断
@Slf4j
public class ReentrantLockDemo3 {
public static void main(String[] args) throws InterruptedException {
ReentrantLock lock = new ReentrantLock();
Thread t1 = new Thread(() -> {
log.debug("t1启动...");
try {
lock.lockInterruptibly();
try {
log.debug("t1获得了锁...");
} finally {
lock.unlock();
}
} catch (InterruptedException e) {
e.printStackTrace();
log.debug("t1等锁的过程被中断...");
}
}, "t1");
lock.lock();
try {
log.debug("main线程获取了锁...");
t1.start();
//先让线程t1执行
Thread.sleep(2000);
t1.interrupt();
log.debug("t1线程执行中断...");
} finally {
lock.unlock();
}
}
}
运行结果:
[main] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo3 - main线程获取了锁…
[t1] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo3 - t1启动…
[main] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo3 - t1线程执行中断…
DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo3 - t1等锁的过程被中断…
java.lang.InterruptedException
at java.util.concurrent.locks.AbstractQueuedSynchronizer.doAcquireInterruptibly(AbstractQueuedSynchronizer.java:898)
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireInterruptibly(AbstractQueuedSynchronizer.java:1222)
at java.util.concurrent.locks.ReentrantLock.lockInterruptibly(ReentrantLock.java:335)
at com.tuling.jucdemo.lock.ReentrantLockDemo3.lambda$main$0(ReentrantLockDemo3.java:22)
at java.lang.Thread.run(Thread.java:745)
2.1.4 锁超时
@Slf4j
public class ReentrantLockDemo4 {
public static void main(String[] args) throws InterruptedException {
ReentrantLock lock = new ReentrantLock();
Thread t1 = new Thread(() -> {
log.debug("t1启动......");
try {
if (!lock.tryLock(1, TimeUnit.SECONDS)) {
log.debug("等待1s后获取锁失败,返回");
return;
}
} catch (InterruptedException e) {
e.printStackTrace();
return;
}
try {
log.debug("t1获得了锁");
} finally {
lock.unlock();
}
}, "t1");
lock.lock();
try{
log.debug("main线程获取了锁");
t1.start();
//先让线程t1执行
Thread.sleep(2000);
}finally {
lock.unlock();
}
}
}
运行结果:
[main] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo4 - main线程获取了锁
[t1] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo4 - t1启动…
[t1] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo4 - 等待1s后获取锁失败,返回
2.1.5 非公平
@Slf4j
public class ReentrantLockDemo5 {
public static void main(String[] args) throws InterruptedException {
//ReentrantLock lock = new ReentrantLock();//默认是非公平 目的是提高效率 会出现插队的情况
ReentrantLock lock = new ReentrantLock(true); //公平 后创建的线程会老老实实排队 不会出现插队的情况
for (int i = 0; i < 500; i++) {
new Thread(()->{
lock.lock();
try{
try {
Thread.sleep(10);
}catch (Exception e){
e.printStackTrace();
}
log.debug(Thread.currentThread().getName() + " running...... ");
}finally {
lock.unlock();
}
},"t" + i).start();
}
//1s之后去竞争
Thread.sleep(1000);
for (int i = 0; i < 500; i++) {
new Thread(()->{
lock.lock();
try{
log.debug(Thread.currentThread().getName() + " ruunning..... ");
}finally {
lock.unlock();
}
},"强行插入" + i).start();
}
}
}
2.1.6 条件变量
@Slf4j
public class ReentrantLockDemo6 {
private static ReentrantLock lock = new ReentrantLock();
private static Condition cigCon = lock.newCondition();
private static Condition takeCon = lock.newCondition();
private static boolean hashcig = false;
private static boolean hastakeout = false;
//送烟
public void cigratee() {
lock.lock();
try {
while (!hashcig) {
log.debug("没有烟,休息一会");
try {
cigCon.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
log.debug("有烟了,干活");
} finally {
lock.unlock();
}
}
//送外卖
public void takeOut(){
lock.lock();
try{
while (!hastakeout){
log.debug("没有饭,休息一会");
try {
takeCon.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
log.debug("有饭了,干活");
}finally {
lock.unlock();
}
}
public static void main(String[] args) {
ReentrantLockDemo6 test = new ReentrantLockDemo6();
new Thread(()->{
test.cigratee();
}).start();
new Thread(()->{
test.takeOut();
}).start();
//唤醒送烟的线程
new Thread(()->{
lock.lock();
try{
hashcig = true;
log.debug("唤醒送烟的等待线程");
cigCon.signal();
}finally {
lock.unlock();
}
},"t1").start();
//唤醒送外卖的等待线程
new Thread(()->{
lock.lock();
try {
hastakeout = true;
log.debug("唤醒送外卖的等待线程");
takeCon.signal();
} finally { // ctrl + alt + t
lock.unlock();
}
},"t2").start();
}
}
运行结果:
[Thread-1] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo6 - 没有饭,休
[t1] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo6 - 唤醒送烟的等待线程
[Thread-0] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo6 - 有烟了,干活
[t2] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo6 - 唤醒送外卖的等待线程
[Thread-1] DEBUG com.tuling.jucdemo.lock.ReentrantLockDemo6 - 有饭了,干活
2.1.7 ReentrantLock与Synchronized区别
(1).synchronized是JVM层次的锁实现,ReentrantLock是JDK层次的锁实现;
(2).synchronized的锁状态是无法在代码中直接判断的,但是ReentrantLock可以通过ReentrantLock#isLocked判断;
(3).synchronized是非公平锁,ReentrantLock是可以是公平也可以是非公平的;
(4).synchronized是不可以被中断的,而ReentrantLock#lockInterruptibly方法是可以被中断的;
(5).在发生异常时synchronized会自动释放锁,而ReentrantLock需要开发者在finally块中显示释放锁;
(6).ReentrantLock获取锁的形式有多种:如立即返回是否成功的tryLock(),以及等待指定时长的获取,更加灵活;
(7).synchronized在特定的情况下对于已经在等待的线程是后来的线程先获得锁(回顾一下sychronized的唤醒策略),而ReentrantLock对于已经在等待的线程是先来的线程先获得锁;
三、ReentrantLock源码分析
关注点:
(1).ReentrantLock加锁解锁的逻辑
(2).公平和非公平,可重入锁的实现
(3).线程竞争锁失败入队阻塞逻辑和获取锁的线程释放锁唤醒阻塞线程竞争锁的逻辑实现 ( 设计的精髓:并发场景下入队和出队操作)
https://www.processon.com/view/link/6191f070079129330ada1209