0

0

如何在Java中理解AQS(AbstractQueuedSynchronizer)_并发框架的核心同步器原理解析

P粉602998670

P粉602998670

发布时间:2026-03-12 04:31:11

|

425人浏览过

|

来源于php中文网

原创

aqs是抽象模板类,需继承并重写tryacquire/tryrelease等钩子方法来定义同步语义;state为volatile int,含义由子类解释,必须用cas操作修改;线程挂起/唤醒使用locksupport而非wait/notify;队列是带头结点的双向链表,取消节点设为cancelled并逻辑隔离。

如何在java中理解aqs(abstractqueuedsynchronizer)_并发框架的核心同步器原理解析

为什么 AQS 不是直接用的类,而是被继承的模板?

因为 AQS 本身不实现具体同步语义,它只提供一套「排队 + 状态管理 + 线程唤醒」的骨架。你看到的 ReentrantLockSemaphoreCountDownLatch 都是它的子类,靠重写 tryAcquiretryRelease 这些钩子方法来定义“什么算获取成功”“怎么才算释放干净”。

常见错误现象:new AQS() 编译报错 —— 它是 abstract 类,不能实例化;有人试图直接调用 acquire() 却没重写 tryAcquire(),结果抛 UnsupportedOperationException

  • 所有同步逻辑必须落在子类对 tryAcquire/tryRelease 等 protected 方法的实现里
  • state 字段是核心状态变量,用 compareAndSetState 操作,不是普通赋值
  • 子类通常把 AQS 声明为内部类(如 ReentrantLock.Sync),避免暴露同步细节

state 字段到底存什么?不同同步器用法差异在哪?

state 是一个 volatile int,但含义完全由子类解释:在 ReentrantLock 里是重入次数,在 Semaphore 里是剩余许可数,在 CountDownLatch 里是倒计数值。它不是“锁标志位”,更不是布尔开关。

容易踩的坑:setState(1) 直接赋值会破坏 CAS 安全性;或者在 tryAcquire 中仅判断 state == 0,却忽略可重入场景(比如没检查当前线程是否已持有锁)。

立即学习Java免费学习笔记(深入)”;

免费语音克隆
免费语音克隆

这是一个提供免费语音克隆服务的平台,用户只需上传或录制一段 5 秒以上的清晰语音样本,平台即可生成与用户声音高度一致的 AI 语音克隆。

下载
  • 修改 state 必须用 compareAndSetStategetAndIncrement 等原子操作
  • ReentrantLock 的非公平模式会在 tryAcquire 中先抢一次,失败才进队列;公平模式则跳过这步,直接排队
  • CountDownLatchstate 只减不增,且 await() 不修改 state,只挂起线程

线程挂起和唤醒靠什么?为什么不是 wait/notify

AQS 底层用的是 LockSupport.park()LockSupport.unpark(),不是对象监视器机制。这意味着它不依赖 synchronized 块,也不受 wait() 必须在同步块中调用的限制。

典型问题:调试时发现线程一直 WAITINGpark(),但没人 unpark() —— 往往是因为 release() 逻辑没触发,或 unparkSuccessor 找不到后继节点(比如节点被取消但没清理干净)。

  • park() 挂起当前线程,unpark(Thread) 唤醒指定线程,两者无配对要求,可提前唤醒
  • 队列中的节点用 Thread 字段保存等待线程,waitStatus 标记是否取消(CANCELLED = 1)或需唤醒(SIGNAL = -1
  • 不要在 tryAcquire 中调用 park() —— acquire() 方法内部已封装了入队 + park 流程

CLH 队列是链表还是真正的 CLH?节点取消时会发生什么?

AQS 的队列是简化版 CLH(Craig, Landin, and Hagersten),实际是带头结点的双向链表,不是原始论文里的“隐式前驱”单向结构。头结点代表正在运行的线程,其余节点代表等待线程。

最易被忽略的细节:节点取消(比如 acquireInterruptibly 被中断)时,cancelAcquire() 会把该节点的 waitStatus 设为 CANCELLED,并跳过它重新连接前后节点 —— 但这个过程不立即删除节点,只是逻辑隔离。后续 shouldParkAfterFailedAcquire 会跳过所有 CANCELLED 前驱。

  • 节点入队用 enq(),首次插入会初始化空头结点,这是唯一可能自旋 CAS 的地方
  • pred.next = nodenode.prev = pred 不是原子的,所以需要循环校验
  • 取消节点不会从内存清除,GC 只能靠前后引用断开后自然回收,大量取消可能导致队列“虚胖”
事情说清了就结束。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

611

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

334

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

75

2025.10.23

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

765

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

99

2025.12.01

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

9

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

22

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号