源码解读--AbstractQueuedSynchronizer讲解之Reentrantlock实现

本文主要是介绍源码解读--AbstractQueuedSynchronizer讲解之Reentrantlock实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一.锁

  说起AbstractQueuedSynchronizer(传说中的AQS),可能有些同学不知道,但是说到ReentrantLock,CountDownLatch,Seamphore,大家可能用用过吧。他们都是用锁来实现了,而锁里面又分独占锁和分享锁。具体锁的种类请看文章。ReentrantLock是独占锁,而CountDownLatch,Seamphore是分享锁。锁里面还有一个重要的分类:公平锁和非公平锁。java源码系列。

公平锁:线程一个一个排队,确保先来的那个等待的线程最早执行。

非公平锁:不能保证先来的那个等待的线程最早执行。

二.AQS

AQS是一个抽象类,经常被使用的是它的子类Sync,会具体实现AQS里面一些方法。AQS里面主要有三个变量。

private transient volatile Node head;  //队列头结点
private transient volatile Node tail;  //队列尾
private volatile int state;  //同步的状态

用了一个双向的链表来存储当前线程信息。大致示意图如下



原理介绍:

以ReentrantLock为例,他的初始状态state=0,当有线程获取锁的时候,state加1,接着后面还有线程过来获取这个锁,就需要在等待队列里面等待。如果之前的线程释放锁,那么后面等待队列里面的线程就可以获取到锁,执行任务。如果期间有中断,这个线程也会被终止掉。

三.源码实现

这里主要介绍以ReentrantLock为基础介绍AQS。

3.1 ReentrantLock锁初始化

public ReentrantLock() {  //默认是new一个非公平的锁,线程之间需要按顺序排队,效率会高一点sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {  //通过true和false来指定是否创建一个公平锁sync = fair ? new FairSync() : new NonfairSync();
}

3.2ReentrantLock 获取锁

public void lock() {  //获取锁sync.lock();  //默认用的是非公平锁,那我们就先来看看非公平锁的实现
}
final void lock() {if (compareAndSetState(0, 1))   //如果还没有线程获取到资源,就将当前线程设置到独占锁setExclusiveOwnerThread(Thread.currentThread());
    else
        acquire(1);  //如果已经有线程占用了资源,其他线程需要排队
}
 
public final void acquire(int arg) { //尝试着去获取资源,如果没有获取到,就会把当前线程封装成一个节点放到同步队列里面,如果添加队列成功,就获取队列里面的资源,如果期间发生了中断,就将当前线程中断掉if (!tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg))selfInterrupt(); //如果线程在阻塞的过程中发生了中断,自身线程也需要中断
}
protected final boolean tryAcquire(int acquires) {  return nonfairTryAcquire(acquires);
}
final boolean nonfairTryAcquire(int acquires) {final Thread current = Thread.currentThread();  //获取当前线程
    int c = getState();  //获取同步状态
    if (c == 0) {  //如果锁获取资源if (compareAndSetState(0, acquires)) { //设置资源的状态setExclusiveOwnerThread(current);  //将当前线程设置为独占模式
            return true;
        }}else if (current == getExclusiveOwnerThread()) {  //检查当前线程是独占模式,如果成立的话,这里可以是线程重入int nextc = c + acquires;   //资源增加acquires个
        if (nextc < 0) // 资源添加的超出int的范围,抛出越界异常
            throw new Error("Maximum lock count exceeded");
        setState(nextc);  //设置资源的状态
        return true;
    }return false;
}

private Node addWaiter(Node mode) {  //线程没有获取到资源,就需要放到同步队列里面Node node = new Node(Thread.currentThread(), mode);  //由当前线程和独占模式创建一个节点
    // Try the fast path of enq; backup to full enq on failure
    Node pred = tail;
    if (pred != null) {  //如果队列不是空node.prev = pred;
        if (compareAndSetTail(pred, node)) { //就将这个node节点添加到队列的尾部pred.next = node;
            return node;
        }}enq(node);  //如果队列为空,就需要初始化这个队列,并将这个ndoe放到队列的尾部
    return node;
}
private Node enq(final Node node) {  //初始队列,并且将node节点入队列for (;;) {  //死循环(自旋)Node t = tail;
        if (t == null) { // 如果节点为空,就初始化一个空的节点当做队头和队尾
            if (compareAndSetHead(new Node()))tail = head;
        } else {  //之前if执行完了,队列里面有了一个空的数据node.prev = t;  //t就是那个之前添加的空的节点
            if (compareAndSetTail(t, node)) { //将node放到队尾t.next = node;
                return t;  //for 循环唯一的出口
            }}}
}
final boolean acquireQueued(final Node node, int arg) {  //获取队列资源,这里面会将获取不到资源的线程阻塞,当有资源可以获取的时候,它会接着执行,但是注意返回的中断的状态boolean failed = true; //结果是否是失败
    try {boolean interrupted = false;  //是否发生了中断
        for (;;) {final Node p = node.predecessor();  //找到node的前驱节点
            if (p == head && tryAcquire(arg)) {  //如果前驱节点是head并且获取资源成功setHead(node);  //将node设置为头节点
                p.next = null; //断链,破引用 以便系统GC
                failed = false;  //执行成功了
                return interrupted;
            }if (shouldParkAfterFailedAcquire(p, node) &&  //ndoe前驱节点不是head,需要返回阻塞的标记parkAndCheckInterrupt()) //如果已经返回了阻塞的标记,则阻塞线程,并且如果返回线程中断的状态interrupted = true;
        }} finally {if (failed)  //未知原因,执行失败,则取消获取资源cancelAcquire(node);
    }
}
private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {int ws = pred.waitStatus;
    if (ws == Node.SIGNAL)  //如果前驱节点的状态是signal,直接返回true,是线程去阻塞
        return true;
    if (ws > 0) {// 线程是取消状态
        do {node.prev = pred = pred.prev;
        } while (pred.waitStatus > 0);  //从后往前过滤取消线程,直到第一个线程不是取消的线程为止
        pred.next = node;
    } else {  //如果不是取消状态 我们就需要将前驱节点的状态设置为signal,到下一次执行该方法是,前驱节点就是signal,返回为true
        compareAndSetWaitStatus(pred, ws, Node.SIGNAL);
    }return false;  
}
private final boolean parkAndCheckInterrupt() { //中断线程 使用LockSupport.park中断当前线程LockSupport.park(this);
    return Thread.interrupted();  //返回的是线程的中断状态,可能这个线程在阻塞的过程中发生过中断
}

小结:到此获取锁已经讲完了。

1.线程先去获取锁,如果可以获取锁,直接返回。

2.如果线程不能获取到锁,就会添加到阻塞队列中,如果队列为空,还需要初始化队列。

3.在线程阻塞的过程中,可能会发生中断,最后可能还需要中断处理一下。

3.3ReentrantLock 释放锁

public void unlock() {  释放锁sync.release(1);
}

public final boolean release(int arg) {if (tryRelease(arg)) {  //是否可以释放资源Node h = head;
        if (h != null && h.waitStatus != 0) //如果队列头节点不为null并且它的状态不是0unparkSuccessor(h);  //唤醒后继节点
        return true;
    }return false;
}
protected final boolean tryRelease(int releases) {  //是否可以释放资源int c = getState() - releases;
    if (Thread.currentThread() != getExclusiveOwnerThread())  //校验如果当前线程不是独占式,那就有问题了,我们知道,我们前面加入的全部都是独占式的throw new IllegalMonitorStateException();
    boolean free = false;
    if (c == 0) {  //如果资源全部放出来了free = true;  
        setExclusiveOwnerThread(null); //将独占式的线程设置为null
    }setState(c); //设置资源的状态
    return free;
}
private void unparkSuccessor(Node node) {
    int ws = node.waitStatus;
    if (ws < 0)  //如果线程没有被取消,设置前驱node节点的的状态是0compareAndSetWaitStatus(node, ws, 0);
    Node s = node.next;
    if (s == null || s.waitStatus > 0) {//如果后继节点是null或者节点是取消状态s = null;
        for (Node t = tail; t != null && t != node; t = t.prev)if (t.waitStatus <= 0)  //从后往前一步步找到后继节点s = t;
    }if (s != null)LockSupport.unpark(s.thread); //唤醒后继节点
}

小结:到此释放锁已经讲完了。

1.先去看线程能不能获取到释放资源的能力

2.如果可以的话就跳过那些已经取消了的线程(是头节点的后继节点里面的线程)唤醒。

四.总结

本文RentrantLock为骨,讲解了非公平锁的实现,借助了底层了AQS,下面来总结一下大致的流程:

初始状态state=0,当有线程获取锁的时候,state加1,接着后面还有线程过来获取这个锁,就需要在等待队列里面等待。同步队列里面头结点获取资源后,会唤醒后继节点去获取锁,当他获取到锁就会执行自己的任务。如果期间有中断,这个线程也会被终止掉。


这篇关于源码解读--AbstractQueuedSynchronizer讲解之Reentrantlock实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1036588

相关文章

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Python FastAPI+Celery+RabbitMQ实现分布式图片水印处理系统

《PythonFastAPI+Celery+RabbitMQ实现分布式图片水印处理系统》这篇文章主要为大家详细介绍了PythonFastAPI如何结合Celery以及RabbitMQ实现简单的分布式... 实现思路FastAPI 服务器Celery 任务队列RabbitMQ 作为消息代理定时任务处理完整

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

C# WinForms存储过程操作数据库的实例讲解

《C#WinForms存储过程操作数据库的实例讲解》:本文主要介绍C#WinForms存储过程操作数据库的实例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、存储过程基础二、C# 调用流程1. 数据库连接配置2. 执行存储过程(增删改)3. 查询数据三、事务处

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义