热门标签 | 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详解


推荐阅读
  • 并发编程 12—— 任务取消与关闭 之 shutdownNow 的局限性
    Java并发编程实践目录并发编程01——ThreadLocal并发编程02——ConcurrentHashMap并发编程03——阻塞队列和生产者-消费者模式并发编程04——闭锁Co ... [详细]
  • 2018-2019学年第六周《Java数据结构与算法》学习总结
    本文总结了2018-2019学年第六周在《Java数据结构与算法》课程中的学习内容,重点介绍了非线性数据结构——树的相关知识及其应用。 ... [详细]
  • Coursera ML 机器学习
    2019独角兽企业重金招聘Python工程师标准线性回归算法计算过程CostFunction梯度下降算法多变量回归![选择特征](https:static.oschina.n ... [详细]
  • 实用正则表达式有哪些
    小编给大家分享一下实用正则表达式有哪些,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下 ... [详细]
  • 本文介绍如何从字符串中移除大写、小写、特殊、数字和非数字字符,并提供了多种编程语言的实现示例。 ... [详细]
  • 深入解析Java虚拟机(JVM)架构与原理
    本文旨在为读者提供对Java虚拟机(JVM)的全面理解,涵盖其主要组成部分、工作原理及其在不同平台上的实现。通过详细探讨JVM的结构和内部机制,帮助开发者更好地掌握Java编程的核心技术。 ... [详细]
  • 深入解析SpringMVC核心组件:DispatcherServlet的工作原理
    本文详细探讨了SpringMVC的核心组件——DispatcherServlet的运作机制,旨在帮助有一定Java和Spring基础的开发人员理解HTTP请求是如何被映射到Controller并执行的。文章将解答以下问题:1. HTTP请求如何映射到Controller;2. Controller是如何被执行的。 ... [详细]
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 本文深入探讨了SQL数据库中常见的面试问题,包括如何获取自增字段的当前值、防止SQL注入的方法、游标的作用与使用、索引的形式及其优缺点,以及事务和存储过程的概念。通过详细的解答和示例,帮助读者更好地理解和应对这些技术问题。 ... [详细]
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • 我有一个SpringRestController,它处理API调用的版本1。继承在SpringRestControllerpackagerest.v1;RestCon ... [详细]
  • 在尝试使用C# Windows Forms客户端通过SignalR连接到ASP.NET服务器时,遇到了内部服务器错误(500)。本文将详细探讨问题的原因及解决方案。 ... [详细]
  • 深入解析动态代理模式:23种设计模式之三
    在设计模式中,动态代理模式是应用最为广泛的一种代理模式。它允许我们在运行时动态创建代理对象,并在调用方法时进行增强处理。本文将详细介绍动态代理的实现机制及其应用场景。 ... [详细]
  • 本文深入探讨了 Java 中 LocalTime 类的 isSupported() 方法,包括其功能、语法和使用示例。通过具体的代码片段,帮助读者理解如何检查特定的时间字段或单位是否被 LocalTime 类支持。 ... [详细]
  • 本题要求在一组数中反复取出两个数相加,并将结果放回数组中,最终求出最小的总加法代价。这是一个经典的哈夫曼编码问题,利用贪心算法可以有效地解决。 ... [详细]
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社区 版权所有