返回顶部
首页 > 资讯 > 后端开发 > Python >教你Java中的Lock锁底层AQS到底是如何实现的
  • 350
分享到

教你Java中的Lock锁底层AQS到底是如何实现的

2024-04-02 19:04:59 350人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

目录前言加锁释放锁总结前言 相信大家对Java中的Lock锁应该不会陌生,比如ReentrantLock,锁主要是用来解决解决多线程运行访问共享资源时的线程安全问题。那你是不是很好奇

前言

相信大家对Java中的Lock锁应该不会陌生,比如ReentrantLock,锁主要是用来解决解决多线程运行访问共享资源时的线程安全问题。那你是不是很好奇,这些Lock锁api是如何实现的呢?本文就是来探讨一下这些Lock锁底层的AQS(AbstractQueuedSynchronizer)到底是如何实现的。

本文是基于ReentrantLock来讲解,ReentrantLock加锁只是对AQS的api的调用,底层的锁的状态(state)和其他线程等待(node双向链表)的过程其实是由AQS来维护的

加锁

我们先来看看加锁的过程,先看源码,然后模拟两个线程来加锁的过程。

上图是ReentrantLock的部分实现。里面有一个Sync的内部类的实例变量,这个Sync内部类继承自AQS,Sync子类就包括公平锁和非公平锁的实现。说白了其实ReentrantLock是通过Sync的子类来实现加锁。

我们就来看一下Sync的非公平锁的实现NonfairSync。

重写了它的lock加锁方法,在实现中因为是非公平的,所以一进来会先通过cas尝试将AQS类的state参数改为1,直接尝试加锁。如果尝试加锁失败会调用AQS的acquire方法继续尝试加锁。

假设这里有个线程1先来调用lock方法,那么此时没有人加锁,那么就通过CAS操作,将AQS中的state中的变量由0改为1,代表有人来加锁,然后将加锁的线程设置为自己如图。

那么此时有另一个线程2来加锁,发现通过CAS操作会失败,因为state已经被设置为1了,线程线程2就会设置失败,那么此时就会走else,调用AQS的acquire方法继续尝试加锁。

进入到acquire会先调用tryAcquire再次尝试加锁,而这个tryAcquire方法AQS其实是没有什么实现的,会调用到NonfairSync里面的tryAcquire,而tryAcquire实际会调用到Sync内部类里面的nonfairTryAcquire非公平尝试加锁方法。

先获取锁的状态,判断锁的状态是不是等于0,等于0说明没人加锁,可以尝试去加,如果被加锁了,就会走else if,else if会判断加锁的线程是不是当前线程,是的话就给state 加 1,代表当前线程加了2次锁,就是可重入锁的意思(所谓的可重入就是代表一个线程可以多次获取到锁,只是将state 设置为多次,当线程多次释放锁之后,将state 设置为0才代表当前线程完全释放了锁)。

这里所有的条件假设都不成立。也就是线程2尝试加锁的时候,线程1并没有释放锁,那么这个方法就会返回false。

接下来就会走到addWaiter方法,这个方法很重要,就是将当前线程封装成一个Node,然后将这个Node放入双向链表中。addWaiter先根据指定模式创建指定的node节点,因为ReentrantLock是独占模式,所以传进去的EXCLUSIVE,这里通过当前线程和模式传入,初始化一个双向node节点,获取最后一个节点,根据最后一个节点是否存在来操作当前节点的父级。如果尾节点不存在会去调用enq去初始化

放入链表中之后如图。

然后调用acquireQueued方法

这个方法一进来也会尝试将当前节点去加锁,然后如果加锁成功就将当前节点设置为头节点,最后将当前线程中断,等待唤醒。

线程2进来的时候,刚好线程2的前一个节点是头节点,但是不巧的是调用tryAcquire方法,还是失败,那么此时就会走shouldParkAfterFailedAcquire方法,这个方法是在线程休眠之前调用的,很重要,我们来看看干了什么事。

判断当前节点的父级节点的状态,如果父级状态是-1,则代表当前线程可以被唤醒了。如果父级的状态为取消状态(什么叫非取消状态,就是tryLock方法等待了一些时间没获取到锁的线程就处于取消状态)就跳过父级,寻找下一个可以被唤醒的父级,然后绑定上节点关系,最后将父级的状态更改为-1。也就说,线程(Node)加入队列之后,如果没有获取到锁,在睡眠之前,会将当前节点的前一个节点设置为非取消状态的节点,然后将前一个节点的waitStatus设置为-1,代表前一个节点在释放锁的时候需要唤醒下一个节点。这一步骤主要是防止当前休眠的线程无法被唤醒。这一切设置成功之后,就会返回true。

接下来就会调用parkAndCheckInterrupt

,这个方法内部调用LockSupport.park方法,此时当前线程就会休眠。

到这一步线程2由于没有获取到锁,就会在这里休眠等待被唤醒。

来总结一下加锁的过程。

线程1先过来,发现没人加锁,那么此时就会加上锁。此时线程2过来,在线程2加锁的过程中,线程1始终没有释放锁,那么线程2就不会加锁成功(如果在线程2加锁的过程中线程1始终释放锁,那么线程2就会加锁成功),线程2没有加锁成功,就会将自己当前线程加入等待队列中(如果没有队列就先初始化一个),然后设置前一个节点的状态,最后通过LockSupport.park方法,将自己这个线程休眠。

如果后面还有线程3,线程4等等诸多的先过来,那么这些线程都会按照前面线程2的步骤,将自己插入链表后面再休眠。

释放锁

ok,说完加锁的过程之后,我们来看看释放锁干了什么。

ReentrantLock的unlock其实是调用AQS的release方法,我们直接进入release方法,看看是如何实现的

进入tryRelease方法,看一下Sync的实现

其实很简单,就是判断锁的状态,也就是加了几次锁,然后减去释放的,最后判断释放之后,锁的状态是不是0(因为可能线程加了多次锁,所以得判断一下),是的话说明当前这个锁已经释放完了,然后将占有锁的线程设置为null,然后返回true,

然后就会走接下来的代码。

就是判断当前链表头节点是不是需要唤醒队列中的线程。如果有链表的话,头结点的waitStatus肯定不是0,因为线程休眠之前,会将前一个节点的状态设置为-1,上面加锁的过程中有提到过。

接下来就会走unparkSuccessor方法,successor代表继承者的意思,见名知意,这个方法其实就会唤醒当前线程中离头节点最近的没有状态为非取消的线程。然后调用LockSupport.unpark,唤醒等待的线

然后线程就会从阻塞的那里苏醒过来,继续尝试获取锁。

我再次贴出这段代码。

获取到锁之后,就将头节点设置成自己。

对应我们的例子,就是线程1释放锁之后,就会唤醒在队列中线程2,先成2获取到锁之后,就会将自己前一个节点(也就是头节点)从链表中移除,将自己设置成头节点。该方法就会跳出死循环。

到这里,释放锁的过程就讲完了,其实很简单,就是当线程完完全全释放了锁,会唤醒当前链表中的没有取消的,离头结点最近的节点(一般就是链表中的第二个节点),然后被唤醒的节点就会获取到锁,将头节点设置为自己。

总结

相信看完这篇文章,大家对AQS的底层有了更深层次的了解。AQS其实就是内部维护一个锁的状态变量state和一个双向链表,加锁成功就将state的值加1,加锁失败就将自己当前线程放入链表的尾部,然后休眠,等待其他线程完完全全释放锁之后将自己唤醒,唤醒之后会尝试加锁,加锁成功就会执行业务代码了。

到此这篇关于教你Java中的Lock锁底层AQS到底是如何实现的的文章就介绍到这了,更多相关Java Lock锁AQS实现内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 教你Java中的Lock锁底层AQS到底是如何实现的

本文链接: https://lsjlt.com/news/150000.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 教你Java中的Lock锁底层AQS到底是如何实现的
    目录前言加锁释放锁总结前言 相信大家对Java中的Lock锁应该不会陌生,比如ReentrantLock,锁主要是用来解决解决多线程运行访问共享资源时的线程安全问题。那你是不是很好奇...
    99+
    2024-04-02
  • java底层AQS实现类ReentrantLock锁的构成及源码解析
    目录引导语1、类注释2、类结构3、构造器 4、Sync 同步器4.1、nonfairTryAcquire 4.2、tryRelease5、FairSync 公平锁...
    99+
    2024-04-02
  • Golang中的Slice底层如何实现
    这篇“Golang中的Slice底层如何实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Golang中的Slice底层如何...
    99+
    2023-07-05
  • sql中的sum底层是怎么实现的
    sql 中 sum 的底层实现原理包括:准备阶段:分配内存缓冲区、获取符合条件的行。累加阶段:将每一行的列值添加到累加器(内存变量)中。优化阶段:使用数据结构优化遍历,跳过特殊值。结果阶...
    99+
    2024-05-09
  • Java中SynchronousQueue的底层实现原理剖析
    目录1. SynchronousQueue用法2. SynchronousQueue应用场景3. SynchronousQueue源码解析3.1 SynchronousQueue类属...
    99+
    2022-11-21
    Java SynchronousQueue实现 Java SynchronousQueue原理 Java SynchronousQueue
  • 详解如何实现SpringBoot的底层注解
    目录一、@Configuration注解二、@Import注解导入组件三、@Conditional注解条件装配四、@ImportResource注解导入Spring配置文件五、@Co...
    99+
    2024-04-02
  • 如何实现PHP底层的异步编程
    如何实现PHP底层的异步编程,需要具体代码示例在传统的编程模型中,PHP是一种基于线程的同步编程语言,即每个请求都会在服务端被依次处理,直到一个请求的处理完成后才会继续处理下一个请求。然而,随着互联网应用的日益复杂和访问量的增加,这种同步模...
    99+
    2023-11-09
    PHP异步编程 PHP底层编程 实现PHP异步
  • 如何实现PHP底层的并发处理
    如何实现PHP底层的并发处理,需要具体代码示例在Web开发过程中,往往需要处理大量的并发请求,如果不采用并发处理的方式,会造成响应时间过长、服务器压力过大等问题。PHP是一种面向Web开发的语言,自带的多线程支持比较弱,但是可以通过其他方式...
    99+
    2023-11-09
    PHP并发编程 PHP异步编程 多线程 PHP
  • 如何实现PHP底层的负载均衡
    如何实现PHP底层的负载均衡负载均衡是指在分布式系统中将工作负载均匀地分配给多个计算资源,以提高系统的并发处理能力和可靠性。在PHP应用程序中,实现负载均衡可以有效地提高系统的性能和可扩展性。本文将介绍如何利用PHP底层技术实现负载均衡,并...
    99+
    2023-11-08
    底层 负载均衡 编程关键词:PHP
  • redis底层数据结构如何实现的
    Redis 底层数据结构的实现 redis 是一种内存中的数据结构存储,它使用高效的数据结构来实现各种数据类型。这些底层数据结构包括: 1. 哈希表(Hash Table) 哈希表用于存...
    99+
    2024-06-12
    redis 键值对
  • 一篇文章教你掌握python数据类型的底层实现
    目录1. 列表1.1 复制1.2 列表的底层实现 - 浅拷贝1.3 浅拷贝 - 示例1. 新增元素2. 修改元素3. 列表型元素4. 元组型元素5. 字典型元素6. 小结1.4 列表...
    99+
    2024-04-02
  • MySQL中索引的底层实现原理是什么
    本篇文章为大家展示了MySQL中索引的底层实现原理是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。  MySQL索引底层实现原理  MySQL官方对索引的定义为...
    99+
    2024-04-02
  • 如何实现PHP底层的分布式缓存
    如何实现PHP底层的分布式缓存随着互联网和大数据时代的到来,对于系统的性能和响应时间要求越来越高。分布式缓存作为提高系统性能的一种重要方式,被广泛应用于各种Web应用中。本文将介绍如何使用PHP底层来实现分布式缓存,并提供具体的代码示例。一...
    99+
    2023-11-08
    PHP分布式 缓存实现 底层技术
  • 如何实现PHP底层的高并发处理
    如何实现PHP底层的高并发处理引言:随着互联网的迅猛发展,高并发的应用需求也越来越多。在PHP这样的脚本语言中,要实现高并发处理需要一些特殊的技巧和方法。本文将介绍如何利用PHP底层的一些特性来实现高并发处理,并附带具体的代码示例。一、使用...
    99+
    2023-11-08
    并发编程 PHP高并发 底层处理
  • 如何实现PHP底层的分布式计算
    如何实现PHP底层的分布式计算随着互联网的快速发展,分布式计算变得越来越重要。而对于PHP开发者来说,实现PHP底层的分布式计算是一个有挑战性的任务。本文将介绍如何使用PHP进行分布式计算,并提供一些具体的代码示例。分布式计算是将一个复杂的...
    99+
    2023-11-08
    底层 PHP 分布式计算
  • java 中的HashMap的底层实现和元素添加流程
    目录HashMap 底层实现HashMap 插入流程为什么要将链表转红黑树?哈希算法实现总结前言: HashMap 是使用频率最高的数据类型之一,同时也是面试必问的问题之一,尤其是它...
    99+
    2024-04-02
  • java中hashmap的底层数据结构与实现原理
    目录Hash结构HashMap实现原理为何HashMap的数组长度一定是2的次幂?重写equals方法需同时重写hashCode方法总结Hash结构 HashMap根据名称可知,其实...
    99+
    2024-04-02
  • 如何实现Python底层技术的线程管理
    抱歉,我在本平台上无法提供直接的代码示例。是否有其他方面的信息我可以帮助你呢?...
    99+
    2023-11-08
    实现方法 线程管理 Python底层技术
  • 如何实现Python底层技术的数据结构
    如何实现Python底层技术的数据结构数据结构是计算机科学中非常重要的一部分,它用于组织和存储数据,以便能够高效地操作和访问数据。Python作为一种高级编程语言,提供了丰富的内置数据结构,如列表、元组、字典等,但有时候我们也需要实现一些底...
    99+
    2023-11-09
    技术实现 底层实现 Python数据结构
  • 如何实现PHP底层的定时任务调度
    实现PHP底层的定时任务调度文章摘要:无论是网站还是后台任务,在很多场景下都需要定时执行某些特定的操作,如发送邮件、数据备份等。PHP作为一种流行的开发语言,本身并没有内置的定时任务调度功能,但我们可以借助一些工具或者编写代码来实现定时任务...
    99+
    2023-11-08
    PHP定时任务 调度器 底层操作
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作