并发能力夯实:ReentrantLock 源码分析!

文章内容已经收录在《高级技术专家成长笔记》,欢迎订阅专栏!

从原理出发,直击面试难点,实现更高维度的降维打击!

目录

  • [ReentrantLock 源码分析]
    • [ReentrantLock 源码结构]
    • [如何基于 AQS 扩展加锁功能?]
    • [ReentrantLock 如何实现公平锁?]
    • [ReentrantLock 如何实现解锁?]

ReentrantLock 源码分析

ReentrantLock 是 JDK1.5 开始提供的可重入锁,用于实现线程间互斥操作,有公平锁和非公平锁两种获取锁的方式。

AQS 提供了 CLH 队列锁来进行线程的阻塞等待和唤醒的功能。

ReentrantLock 底层基于 AQS 的线程阻塞、唤醒功能,实现了线程加锁,比如 tryLock()tryLock(timeout)lockInterruptibly() 功能。

ReentrantLock 源码结构

概括来说的话 ReentrantLock 就是在 AQS 外边套了一层皮,实现了加锁功能。

ReentrantLock 主要由 3 个类组成:

  • Sync :继承自 AbstractQueuedSynchronizer ,获得了线程阻塞、唤醒的能力,并扩展了获取锁、释放锁的功能。
  • NonfairSync :继承自 Sync ,提供了非公平获取锁的功能。
  • FairSync :继承自 Sync ,提供了公平获取锁的功能。

结构如下:

JAVA 复制代码
public class ReentrantLock implements Lock, java.io.Serializable {
    // 1、同步器
    abstract static class Sync extends AbstractQueuedSynchronizer {}
    // 2、非公平同步器
    static final class NonfairSync extends Sync {}
    // 3、公平同步器
    static final class FairSync extends Sync {}
}

如何基于 AQS 扩展加锁功能?

AQS 作为一个底层工具,仅提供了线程阻塞和唤醒的能力,并没有加锁能力,ReentrantLock 基于 AQS 的能力,实现了加锁、解锁等一系列功能。

接下来从 ReentrantLock 的加锁方法入手,来看它是如何基于 AQS 的能力来进行扩展的。

ReentrantLock 使用起来比较简单,只需要创建一个 ReentrantLock 对象,并且调用 lock() 方法就可以完成加锁,如下:

JAVA 复制代码
public class ReentrantLockDemo {
    public static void main(String[] args) {
        ReentrantLock lock = new ReentrantLock();
        lock.lock();
    }
}

接下来进入到 ReentrantLock # lock() 方法:

JAVA 复制代码
public void lock() {
    sync.lock();
}

可以看到在内部调用了 sync # lock() 方法,使用 Sync 同步器来完成加锁。

上边讲了 Sync 分为 NonfairSyncFairSync ,这里以 NonfairSync 为例进行讲解。

通过 sync # lock() 方法加锁,最终需要借助 AQS 的能力完成,调用链走到 AbstractQueuedSynchronizer # acquire() 方法内部,如下:

JAVA 复制代码
public abstract class AbstractQueuedSynchronizer extends AbstractOwnableSynchronizer implements java.io.Serializable {
    public final void acquire(int arg) {
        if (!tryAcquire(arg) &&
            acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
            selfInterrupt();
    }
}

AbstractQueuedSynchronizer # acquire() 方法内部,核心有 3 个操作:

  • tryAcquire() :尝试获取锁。
  • addWaiter(Node.EXCLUSIVE), arg) :如果获取锁失败,会将当前线程封装为 Node 节点加入到 AQS 的 CLH 队列中。
  • acquireQueued() :对线程进行阻塞、唤醒,并不断调用 tryAcquire() 方法去尝试获取锁。

重点来了!

tryAcuquire() 方法在 AQS 提供的默认实现直接抛出异常,如下所示,具体功能留给子类去实现。

JAVA 复制代码
public abstract class AbstractQueuedSynchronizer extends AbstractOwnableSynchronizer implements java.io.Serializable {
    protected boolean tryAcquire(int arg) {
        throw new UnsupportedOperationException();
    }
}

因此在 ReentrantLock 内部,NonfairSyncFairSync 就重写了 tryAcquire() 方法,实现了 非公平锁公平锁

NonfairSynctryAcquire() 中,最终调用了 Sync # nonfairTryAcquire() 方法, 获取非公平锁的功能由 Sync 来支撑。

Sync # nonfairTryAcquire() 代码如下:

JAVA 复制代码
abstract static class Sync extends AbstractQueuedSynchronizer {
    // 非公平加锁
    final boolean nonfairTryAcquire(int acquires) {
        final Thread current = Thread.currentThread();
        // 1、获取 AQS 中的 state 状态
        int c = getState();
        // 2、如果 state 为 0,证明锁没有被其他线程占用
        if (c == 0) {
            // 2.1、通过 CAS 对 state 进行更新
            if (compareAndSetState(0, acquires)) {
                // 2.2、如果 CAS 更新成功,就将锁的持有者设置为当前线程
                setExclusiveOwnerThread(current);
                return true;
            }
        }
        // 3、如果当前线程和锁的持有线程相同,说明发生了「锁的重入」
        else if (current == getExclusiveOwnerThread()) {
            int nextc = c + acquires;
            if (nextc < 0) // overflow
                throw new Error("Maximum lock count exceeded");
            // 3.1、将锁的重入次数加 1
            setState(nextc);
            return true;
        }
        // 4、如果锁被其他线程占用,就返回 false,表示获取锁失败
        return false;
	}
}

Sync # nonfairTryAcquire() 方法内部,主要通过两个核心操作去完成加锁:

  • 通过 CAS 更新 state 变量。state == 0 表示锁没有被占用。state > 0 表示锁被占用,此时 state 表示重入次数。
  • 通过 setExclusiveOwnerThread() 设置持有锁的线程。

非公平性体现在哪里?

Sync#nonfairTryAcquire() 方法中,首先会检查 AQS 的 state 状态。如果 state == 0,即锁未被占用,线程会通过 CAS 操作尝试更新 state 的值。如果 CAS 更新成功,则当前线程成功获取锁。

非公平性 的体现就在于:每个线程直接通过 CAS 操作去抢占锁,不需要在队列中排队等待。

而公平锁的实现中会先检查是否有线程已经在等待队列中。如果有其他线程在等待,当前线程会返回 false,表示获取锁失败;如果没有线程在等待,线程才会尝试通过 CAS 更新锁的状态。

ReentrantLock 加锁总结:

AQS 提供了线程阻塞、唤醒的能力,并且暴露了 tryAcquire() 方法供子类去实现具体的逻辑。

ReentrantLock 继承了 AQS ,通过重写了 tryAcquire() 方法来实现线程的加锁和锁重入的功能。

为了更加明确 ReentrantLock 是如何进行包装的,这里只讲 ReentrantLock 基于 AQS 所包装的能力,不会去讲 AQS 内部是如何实现线程的阻塞和唤醒。

ReentrantLock 如何实现公平锁?

接下来讲 ReentrantLock 内部 公平锁 的实现原理。

ReentrantLock 内部的公平锁通过 FairSync 来支撑,其内部重写了 tryAcquire() 方法来实现公平锁。

公平锁和非公平锁 只有一行代码的差别 ,代码对比如下:

第 6 行 ,公平锁通过 CAS 更新 state 状态之前,会先通过 hasQueuedPredecessors() 方法去判断队列前边是否已经有等待线程,如果没有等待线程的话,再去执行 CAS 获取锁,以此来实现 公平性

hasQueuedPredecessors() 方法是 AQS 提供的能力,AQS 通过 CLH 队列来存储等待的线程,该方法就是去 CLH 队列中查看是否已经存在等待的线程,代码如下:

JAVA 复制代码
// AbstractQueuedSynchronizer # hasQueuedPredecessors()
public final boolean hasQueuedPredecessors() {
    // 1、获取 tail
    Node t = tail; 
    // 2、获取 head
    Node h = head;
    Node s;
    // 3、判断是否存在等待线程
    return h != t &&
        ((s = h.next) == null || s.thread != Thread.currentThread());
}

hasQueuedPredecessors() 方法中,会先获取 tail ,之后再获取 head ,如果 headtail 不相同,并且 head 后边的第一个线程不是当前线程,则证明有线程在队列中等待。

该方法中,会先执行 t = tail ,再执行 h = head ,赋值顺序和队列的初始化顺序相反,这样做是为了保证并发安全。

因为在 AQSCLH 队列初始化时,会先初始化 head ,再初始化 tail

因此,如果在进入 hasQueuedPredecessors() 方法时,队列还没有初始化,此时执行完 h = head 之后,h 指向 null,假设此时其他线程初始化队列,此时再执行 t = tail ,之后 t 就指向了初始化后的队列,导致 ht 指向的不是同一时刻的节点,该方法的计算结果就不准确了。

AQS 中大量使用了 CAS 无锁操作,给性能带来提升的同时也增加了编程的复杂度。

ReentrantLock 如何实现解锁?

ReentrantLock 的解锁功能是通过内部的 Sync 重写 tryRelease() 方法来实现的,和加锁功能实现原理相似,代码如下:

JAVA 复制代码
abstract static class Sync extends AbstractQueuedSynchronizer {
    protected final boolean tryRelease(int releases) {
        // 1、计算解锁后的 state 值
        int c = getState() - releases;
        if (Thread.currentThread() != getExclusiveOwnerThread())
            throw new IllegalMonitorStateException();
        boolean free = false;
        // 2、如果 state 值为 0,则表明该线程释放锁
        if (c == 0) {
            free = true;
            setExclusiveOwnerThread(null);
        }
        // 3、直接更新 state 值
        setState(c);
        return free;
    }
}

由于加锁的时候已经实现线程互斥了,因此在 解锁 的时候一定是单线程执行的,不需要通过 CAS 控制并发安全。

在解锁的时候,有两种情况:

  • state 值不为 0,表明该线程仍处在锁的重入。
  • state 值为 0,表明该线程释放锁,将持有锁的线程设置为 null。
相关推荐
计算机-秋大田4 分钟前
基于微信小程序的校园失物招领系统设计与实现(LW+源码+讲解)
java·前端·后端·微信小程序·小程序·课程设计
綦枫Maple5 分钟前
Spring Boot(6)解决ruoyi框架连续快速发送post请求时,弹出“数据正在处理,请勿重复提交”提醒的问题
java·spring boot·后端
码至终章41 分钟前
kafka常用目录文件解析
java·分布式·后端·kafka·mq
Mr.Demo.1 小时前
[Spring] Nacos详解
java·后端·spring·微服务·springcloud
梁雨珈1 小时前
PL/SQL语言的图形用户界面
开发语言·后端·golang
智_永无止境1 小时前
Springboot使用war启动的配置
java·spring boot·后端·war
Ciderw2 小时前
MySQL为什么使用B+树?B+树和B树的区别
c++·后端·b树·mysql·面试·golang·b+树
计算机-秋大田2 小时前
基于微信小程序的汽车保养系统设计与实现(LW+源码+讲解)
spring boot·后端·微信小程序·小程序·课程设计
齐雅彤2 小时前
Bash语言的并发编程
开发语言·后端·golang
峰子20122 小时前
B站评论系统的多级存储架构
开发语言·数据库·分布式·后端·golang·tidb