热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

JUC(三):深入解析AQS

本文详细介绍了Java并发工具包中的核心类AQS(AbstractQueuedSynchronizer),包括其基本概念、数据结构、源码分析及核心方法的实现。
一. AbstractQueuedSynchronizer简介

AQS(AbstractQueuedSynchronizer)是Java并发工具包中的一个框架类,用于构建锁和同步器。通过使用AQS,开发者可以轻松地创建各种同步器,如ReentrantLock、Semaphore等。AQS的核心思想是通过一个int类型的成员变量表示同步状态,并使用FIFO队列来管理线程的等待和唤醒。

1. AQS 核心思想

AQS的核心思想是,如果请求的共享资源空闲,则将当前请求的线程设置为有效的工作线程,并锁定资源。如果资源被占用,则需要一套线程阻塞和唤醒机制(CLH队列),即将线程加入到队列中等待资源释放。

AQS将线程封装成CLH锁队列的一个节点(Node)来实现锁的分配。AQS使用一个int成员变量表示同步状态,通过FIFO队列来完成排队,AQS通过CAS操作完成对状态的修改。

2. AQS 对资源的共享方式

AQS定义了两种资源共享方式:

  • Exclusive(独占):只有一个线程能执行,如ReentrantLock。又分为公平锁和非公平锁:
    • 公平锁:按照队列中的排队顺序获取锁。
    • 非公平锁:无视队列顺序,谁抢到就是谁的。
  • Share(共享):多个线程可同时执行,如Semaphore/CountDownLatch。

不同的自定义同步器争用共享资源的方式也不同。自定义同步器在实现时只需要实现共享资源state的获取与释放方式即可,至于具体线程等待队列的维护(如获取资源失败入队/唤醒出队等),AQS已经帮我们实现好了。

二. AbstractQueuedSynchronizer数据结构

AQS的底层数据结构使用CLH队列(一个虚拟的双向队列)。AQS将请求资源的线程封装成CLH锁队列的一个节点来实现锁的分配。

  • Sync Queue:同步队列,使用双向链表,其中head节点主要用于后续的调度。
  • Condition Queue:不是必须的,是一个单向链表,只有使用condition时才会使用此单向链表。

CLH队列示意图

三. AbstractQueuedSynchronizer源码分析

内部类 - Node类

Node类是AQS的内部类,用于表示同步队列中的节点。每个节点包含以下信息:

  • 模式:分为共享模式(SHARED)和独占模式(EXCLUSIVE)。
  • 状态:表示节点的状态,如CANCELLED、SIGNAL、CONDITION、PROPAGATE等。
  • 前驱和后继节点:用于形成双向链表。
  • 线程:节点所对应的线程。

内部类 - ConditionObject类

ConditionObject类实现了Condition接口,提供了条件操作规范,包括await、signal等方法。这些方法用于控制线程的等待和唤醒。

类的属性

AQS类包含了一些重要的属性,如头节点head、尾节点tail、状态state等。这些属性通过Unsafe类进行CAS操作,确保线程安全。

类的核心方法 - acquire方法

acquire方法以独占模式获取资源,忽略中断。具体步骤如下:

  1. 调用tryAcquire方法尝试获取资源,如果成功则返回。
  2. 如果tryAcquire失败,则调用addWaiter方法将当前线程封装成节点并加入同步队列。
  3. 调用acquireQueued方法,使节点在队列中不断尝试获取资源,直到成功或被中断。

类的核心方法 - release方法

release方法以独占模式释放资源。具体步骤如下:

  1. 调用tryRelease方法尝试释放资源,如果成功则继续。
  2. 如果头节点不为空且状态不为0,则调用unparkSuccessor方法唤醒头节点的后继节点。
AbstractQueuedSynchronizer总结

对于AQS的分析,最核心的部分是同步队列(sync queue)的管理。以下是几个关键点:

  • 每个节点都是由前一个节点唤醒。
  • 当节点发现前驱节点是头节点并且尝试获取资源成功,则会轮到该线程运行。
  • condition queue中的节点向sync queue中转移是通过signal操作完成的。
  • 当节点的状态为SIGNAL时,表示后面的节点需要运行。

参考:AQS详解


推荐阅读
  • 本文详细介绍了 PHP 中对象的生命周期、内存管理和魔术方法的使用,包括对象的自动销毁、析构函数的作用以及各种魔术方法的具体应用场景。 ... [详细]
  • 2020年9月15日,Oracle正式发布了最新的JDK 15版本。本次更新带来了许多新特性,包括隐藏类、EdDSA签名算法、模式匹配、记录类、封闭类和文本块等。 ... [详细]
  • 本文介绍了在 Java 编程中遇到的一个常见错误:对象无法转换为 long 类型,并提供了详细的解决方案。 ... [详细]
  • 深入剖析Java中SimpleDateFormat在多线程环境下的潜在风险与解决方案
    深入剖析Java中SimpleDateFormat在多线程环境下的潜在风险与解决方案 ... [详细]
  • POJ 2482 星空中的星星:利用线段树与扫描线算法解决
    在《POJ 2482 星空中的星星》问题中,通过运用线段树和扫描线算法,可以高效地解决星星在窗口内的计数问题。该方法不仅能够快速处理大规模数据,还能确保时间复杂度的最优性,适用于各种复杂的星空模拟场景。 ... [详细]
  • C++ 开发实战:实用技巧与经验分享
    C++ 开发实战:实用技巧与经验分享 ... [详细]
  • MySQL初级篇——字符串、日期时间、流程控制函数的相关应用
    文章目录:1.字符串函数2.日期时间函数2.1获取日期时间2.2日期与时间戳的转换2.3获取年月日、时分秒、星期数、天数等函数2.4时间和秒钟的转换2. ... [详细]
  • 本文节选自《NLTK基础教程——用NLTK和Python库构建机器学习应用》一书的第1章第1.2节,作者Nitin Hardeniya。本文将带领读者快速了解Python的基础知识,为后续的机器学习应用打下坚实的基础。 ... [详细]
  • 零拷贝技术是提高I/O性能的重要手段,常用于Java NIO、Netty、Kafka等框架中。本文将详细解析零拷贝技术的原理及其应用。 ... [详细]
  • 在多线程并发环境中,普通变量的操作往往是线程不安全的。本文通过一个简单的例子,展示了如何使用 AtomicInteger 类及其核心的 CAS 无锁算法来保证线程安全。 ... [详细]
  • com.hazelcast.config.MapConfig.isStatisticsEnabled()方法的使用及代码示例 ... [详细]
  • 在分析和解决 Keepalived VIP 漂移故障的过程中,我们发现主备节点配置如下:主节点 IP 为 172.16.30.31,备份节点 IP 为 172.16.30.32,虚拟 IP 为 172.16.30.10。故障表现为监控系统显示 Keepalived 主节点状态异常,导致 VIP 漂移到备份节点。通过详细检查配置文件和日志,我们发现主节点上的 Keepalived 进程未能正常运行,最终通过优化配置和重启服务解决了该问题。此外,我们还增加了健康检查机制,以提高系统的稳定性和可靠性。 ... [详细]
  • 本文介绍了如何利用Shell脚本高效地部署MHA(MySQL High Availability)高可用集群。通过详细的脚本编写和配置示例,展示了自动化部署过程中的关键步骤和注意事项。该方法不仅简化了集群的部署流程,还提高了系统的稳定性和可用性。 ... [详细]
  • V8不仅是一款著名的八缸发动机,广泛应用于道奇Charger、宾利Continental GT和BossHoss摩托车中。自2008年以来,作为Chromium项目的一部分,V8 JavaScript引擎在性能优化和技术创新方面取得了显著进展。该引擎通过先进的编译技术和高效的垃圾回收机制,显著提升了JavaScript的执行效率,为现代Web应用提供了强大的支持。持续的优化和创新使得V8在处理复杂计算和大规模数据时表现更加出色,成为众多开发者和企业的首选。 ... [详细]
  • 本文深入解析了Java 8并发编程中的`AtomicInteger`类,详细探讨了其源码实现和应用场景。`AtomicInteger`通过硬件级别的原子操作,确保了整型变量在多线程环境下的安全性和高效性,避免了传统加锁方式带来的性能开销。文章不仅剖析了`AtomicInteger`的内部机制,还结合实际案例展示了其在并发编程中的优势和使用技巧。 ... [详细]
author-avatar
溜溜的情歌哆瑞咪
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有