面试基础---深入解析 AQS
深入解析 AQS:从源码到实践,剖析 ReentrantLock 和 Semaphore 的实现
引言
在 Java 并发编程中,AbstractQueuedSynchronizer(AQS)是一个核心框架,它为构建锁和其他同步器提供了基础支持。ReentrantLock 和 Semaphore 是 AQS 的两个典型实现,分别用于实现可重入锁和信号量。本文将从底层源码的角度,深入分析 AQS 的核心机制,并结合 ReentrantLock 和 Semaphore 的实际应用场景,探讨其设计思想与实现细节。
一、AQS 的核心机制
1.1 AQS 的设计思想
AQS 的核心思想是通过一个 FIFO 队列(CLH 队列)来管理线程的排队和唤醒机制,同时结合一个 state 变量来表示同步状态。AQS 提供了两种模式:
- 独占模式:同一时刻只有一个线程可以获取资源(如
ReentrantLock)。 - 共享模式:多个线程可以同时获取资源(如
Semaphore)。
1.2 核心数据结构
1.2.1 state 变量
state 是 AQS 的核心变量,用于表示同步状态。例如:
- 在
ReentrantLock中,state表示锁的重入次数。 - 在
Semaphore中,state表示剩余的许可数。
private volatile int state; // 同步状态
1.2.2 CLH 队列
AQS 使用 CLH 队列(Craig, Landin, and Hagersten 锁队列)来管理等待线程。每个线程被封装为一个 Node 节点,节点中保存了线程的状态(如等待、取消)以及前驱和后继节点的引用。
static final class Node {volatile int waitStatus; // 线程状态volatile Node prev; // 前驱节点volatile Node next; // 后继节点volatile Thread thread; // 等待线程
}
1.3 关键方法
1.3.1 acquire(int arg)
acquire 是获取资源的核心方法。它的主要逻辑如下:
- 调用
tryAcquire尝试获取资源。 - 如果获取失败,将当前线程封装为
Node并加入 CLH 队列。 - 通过自旋和
LockSupport.park()挂起线程,等待唤醒。
public final void acquire(int arg) {if (!tryAcquire(arg) &&acquireQueued(addWaiter(Node.EXCLUSIVE), arg))selfInterrupt();
}
1.3.2 release(int arg)
release 是释放资源的核心方法。它的主要逻辑如下:
- 调用
tryRelease尝试释放资源。 - 如果释放成功,唤醒 CLH 队列中的下一个线程。
public final boolean release(int arg) {if (tryRelease(arg)) {Node h = head;if (h != null && h.waitStatus != 0)unparkSuccessor(h);return true;}return false;
}
二、ReentrantLock 的实现
2.1 公平锁与非公平锁
ReentrantLock 支持公平锁和非公平锁两种模式:
- 公平锁:严格按照 CLH 队列的顺序获取锁。
- 非公平锁:允许插队,新线程可以直接尝试获取锁。
2.1.1 公平锁的实现
公平锁在 tryAcquire 中会检查是否有前驱节点,如果有则放弃获取锁。
protected final boolean tryAcquire(int acquires) {if (getState() == 0 && !hasQueuedPredecessors() &&compareAndSetState(0, acquires)) {setExclusiveOwnerThread(currentThread());return true;}return false;
}
2.1.2 非公平锁的实现
非公平锁在 tryAcquire 中不会检查前驱节点,直接尝试获取锁。
final boolean nonfairTryAcquire(int acquires) {if (getState() == 0 && compareAndSetState(0, acquires)) {setExclusiveOwnerThread(currentThread());return true;}return false;
}
2.2 tryLock() 和 lock() 方法
lock():调用acquire(1),如果获取失败则进入等待队列。tryLock():调用tryAcquire(1),立即返回是否获取成功。
三、Semaphore 的实现
3.1 信号量的许可数管理
Semaphore 通过 state 变量表示剩余的许可数。acquire() 和 release() 方法分别用于获取和释放许可。
3.1.1 acquire()
acquire 方法会减少 state 的值,如果许可不足则进入等待队列。
public void acquire() throws InterruptedException {sync.acquireSharedInterruptibly(1);
}
3.1.2 release()
release 方法会增加 state 的值,并唤醒等待线程。
public void release() {sync.releaseShared(1);
}
四、实际应用场景
4.1 ReentrantLock 的应用
- 高并发环境下的资源竞争控制:如数据库连接池的并发访问。
- 可重入特性:支持同一线程多次获取锁,避免死锁。
4.2 Semaphore 的应用
- 线程池的任务调度:通过信号量限制并发任务数。
- 限流:控制系统的并发请求数,防止资源耗尽。
五、性能优化和注意事项
5.1 性能优化
- 自旋锁:在短时间内通过自旋尝试获取锁,减少线程切换的开销。
- CAS 操作:通过
compareAndSetState实现无锁化的状态更新。
5.2 注意事项
- 避免死锁:确保锁的获取和释放成对出现。
- 合理设置超时:使用
tryLock(long timeout, TimeUnit unit)避免长时间等待。
六、总结与展望
AQS 是 Java 并发包的基石,其设计思想(CLH 队列 + state 变量)为构建高效、灵活的同步器提供了强大支持。ReentrantLock 和 Semaphore 是 AQS 的典型应用,分别解决了独占资源和共享资源的同步问题。
未来,AQS 可能会在以下方面进一步优化:
- 更高效的自旋策略。
- 对 NUMA 架构的更好支持。
- 更灵活的同步模式扩展。
附录
图表:CLH 队列结构
Head -> Node1 -> Node2 -> Node3 -> Tail

关键代码片段
- AQS 的
acquire和release方法。 ReentrantLock的公平锁与非公平锁实现。Semaphore的acquire和release方法。
通过本文的分析,相信读者能够深入理解 AQS 的设计思想及其在 ReentrantLock 和 Semaphore 中的应用,为高并发编程打下坚实基础。
相关文章:
面试基础---深入解析 AQS
深入解析 AQS:从源码到实践,剖析 ReentrantLock 和 Semaphore 的实现 引言 在 Java 并发编程中,AbstractQueuedSynchronizer(AQS)是一个核心框架,它为构建锁和其他同步器提供了基础支持。ReentrantLock 和…...
go 语言中的线程池
使用 goroutine 和 channel Go 语言中并没有直接类似 Java 线程池的内建概念,但它提供了类似的功能,主要通过goroutine和channel来实现并发处理。你可以通过结合这两者来实现一个“线程池”的功能。 在 Go 中,goroutine是轻量级的线程&…...
从 0 到 1,用 Python 构建超实用 Web 实时聊天应用
从 0 到 1,用 Python 构建超实用 Web 实时聊天应用 本文深入剖析如何运用 Python 的 Flask 框架与 SocketIO 扩展,搭建一个功能完备的 Web 实时聊天应用。从环境搭建、前后端代码实现,到最终运行展示,逐步拆解关键步骤࿰…...
AF3 DataPipeline类process_multiseq_fasta 方法解读
AlphaFold3 data_pipeline 模块DataPipeline类的 process_multiseq_fasta 方法用于处理多序列 FASTA 文件,生成 AlphaFold3 结构预测所需的特征,适用于多链复合物的预测。它结合了 Minkyung Baek 在 Twitter 上提出的“AlphaFold-Gap”策略,即通过在多链 MSA 中插入固定长度…...
Vue2+Element实现Excel文件上传下载预览【超详细图解】
目录 一、需求背景 二、落地实现 1.文件上传 图片示例 HTML代码 业务代码 2.文件下载 图片示例 方式一:代码 方式二:代码 3.文件预览 图片示例 方式一:代码 方式二:代码 一、需求背景 在一个愉快的年后ÿ…...
[记录贴] 火绒奇怪的进程保护
最近一次更新火绒6.0到最新版,发现processhacker的结束进程功能无法杀掉火绒的进程,弹窗提示如下: 可能是打开进程时做了权限过滤,火绒注册了两个回调函数如下: 但奇怪的是,在另外一台机器上面更新到最新版…...
【蓝桥杯】每天一题,理解逻辑(1/90)【Leetcode 移动零】
文章目录 题目解析讲解算法原理【双指针算法思路】(数组下标充当指针)如何划分和执行过程大致 代码详情 题目解析 题目链接:https://leetcode.cn/problems/move-zeroes/description/ 题目意思解析 把所有的零移动到数组的末尾保持非零元素的相对顺序 理解了这两层…...
vue js-web-screen-shot浏览器截取其他非全屏窗口界面
网页截屏 js-web-screen-shot 截取其他窗口 显示不全问题 npm 安装 js-web-screen-shot npm install js-web-screen-shot --savejs-web-screen-shot默认截屏是从左下角开始的,修改成左上角开始,然后编辑cropBoxInfo参数宽高进行截取,目前截…...
pycharm远程连接服务器运行pytorch
Linux部署pytorch 背景介绍 不同的开源代码可能需要不同的实验环境和版本,这时候的确体现出Anaconda管理环境的好处了,分别搞一个独立环境方便管理。 有的教程建议选择较旧的版本,但笔者建议在条件允许的情况下安装最新版,本次…...
服务器虚拟化是一种将物理服务器资源(如CPU、内存、存储、网络等)通过软件技术抽象、分割和整合,创建多个独立、隔离的虚拟服务器(虚拟机,VM)的技术。
服务器虚拟化是一种将物理服务器资源(如CPU、内存、存储、网络等)通过软件技术抽象、分割和整合,创建多个独立、隔离的虚拟服务器(虚拟机,VM)的技术。每个虚拟机可以运行不同的操作系统和应用程序,如同独立的物理服务器一样工作。 核心思想 资源池化:将物理服务器的硬…...
java练习(41)
ps:题目来自力扣 最接近的三数之和 给你一个长度为 n 的整数数组 nums 和 一个目标值 target。请你从 nums 中选出三个整数,使它们的和与 target 最接近。 返回这三个数的和。 假定每组输入只存在恰好一个解。 import java.util.Arrays;class Solut…...
关于CanvasRenderer.SyncTransform触发调用的机制
1)关于CanvasRenderer.SyncTransform触发调用的机制 2)小游戏Spine裁剪掉帧问题 3)Dedicated Server性能问题 4).mp4视频放入RT进行渲染的性能分析闭坑指南 这是第421篇UWA技术知识分享的推送,精选了UWA社区的热门话题…...
【计算机网络】OSI模型、TCP/IP模型、路由器、集线器、交换机
一、计算机网络分层结构 计算机网络分层结构 指将计算机网络的功能划分为多个层次,每个层次都有其特定的功能和协议,并且层次之间通过接口进行通信。 分层设计的优势: 模块化:各层独立发展(如IPv4→IPv6,…...
PDF扫描档智能方向识别:多模型投票机制的实践测试 救活古典书籍
2025-02-22 20:10物联全栈123 尊敬的诸位!我是一名物联网工程师。关注我,持续分享最新物联网与AI资讯和开发实战。期望与您携手探寻物联网与 AI 的无尽可能 RAG知识库搭建的过程中,扫描档pdf的支持和准确率一直是个大家都不愿主动提起的事情…...
java23种设计模式-桥接模式
桥接模式(Bridge Pattern)学习笔记 🌟 定义 桥接模式属于结构型设计模式,将抽象部分与实现部分分离,使它们可以独立变化。通过组合代替继承的方式,解决多维度的扩展问题,防止类爆炸。 &#x…...
【11】RUST使用cargo组织crate
文章目录 使用cargo组织crate重导出编译文档生成测试 cargo组织工作空间 TODOcrate.io账号 TODO暂时不看发布crate 使用cargo组织crate 重导出 在模块顶部使用pub use self::重导出,方便使用模块时候直接使用use mod_X::xxx。从而隐藏crate内部模块的结构。方便向…...
springboot 引入前端
前端 打包 npm run build vue.config.js 文件 publicPath 默认建议保持 / publicPath: ‘/’ 后端 目录 粘贴下面目录之一: src/main/resources/static/ src/main/resources/public/ 补充(用的少) server:servlet:context-path: /thirdAdm…...
ubuntu20.04 使用nmcli 连接wifi,并且设置永久连接
在 Ubuntu 20.04 中,你可以使用 nmcli 命令行工具来连接 WiFi 并设置为永久连接。以下是具体步骤: 1. 查看可用的 WiFi 网络 首先,使用以下命令查看可用的 WiFi 网络: nmcli dev wifi 这将列出所有可用的 WiFi 网络及其 SSID。…...
Android-创建mipmap-anydpi-v26的Logo
利用 Android Studio 自动创建 创建新项目:打开 Android Studio,点击 “Start a new Android Studio project” 创建新项目。在创建项目的过程中,当设置Target SDK Version为 26 或更高版本时,Android Studio 会在项目的res目录下…...
轻松搭建:使用Anaconda创建虚拟环境并在PyCharm中配置
一、使用Anaconda创建虚拟环境 1. 安装Anaconda 2..conda常用的命令 3. 创建虚拟环境-以搭建MachineVision为例 4. 激活虚拟环境 5. 安装依赖包 二、PyCharm配置环境 在进行Python项目开发时,合理的环境管理是必不可少的,特别是当你在多个项目中…...
驱动开发系列39 - Linux Graphics 3D 绘制流程(二)- 设置渲染管线
一:概述 Intel 的 Iris 驱动是 Mesa 中的 Gallium 驱动,主要用于 Intel Gen8+ GPU(Broadwell 及更新架构)。它负责与 i915 内核 DRM 驱动交互,并通过 Vulkan(ANV)、OpenGL(Iris Gallium)、或 OpenCL(Clover)来提供 3D 加速。在 Iris 驱动中,GPU Pipeline 设置 涉及…...
结构型模式 - 代理模式 (Proxy Pattern)
结构型模式 - 代理模式 (Proxy Pattern) 代理模式是一种结构型设计模式,它允许通过代理对象来控制对另一个对象(目标对象)的访问。代理对象充当目标对象的接口,客户端通过代理对象间接访问目标对象。 分为两大类 静态代理&#…...
Ubuntu 24.04 登录禁用用户列表
Ubuntu 24.04 登录禁用用户列表 得到登录用户列表状态禁用登录用户列表 得到登录用户列表状态 gsettings get org.gnome.login-screen disable-user-list禁用登录用户列表 gsettings get org.gnome.login-screen disable-user-list true...
MinIO整合SpringBoot实现文件上传、下载
文章目录 配置1. 部署MinIO服务2. 整合SpringBoot 功能实现1. 文件上传2. 文件下载 总结 配置 1. 部署MinIO服务 这里以docker为例: 安装minio命令docker run -p 9000:9000 -p 9001:9001 \ --name minio \ -v /path/to/data:/data \ -e "MINIO_ROOT_USERmin…...
【Python爬虫(90)】以Python爬虫为眼,洞察金融科技监管风云
【Python爬虫】专栏简介:本专栏是 Python 爬虫领域的集大成之作,共 100 章节。从 Python 基础语法、爬虫入门知识讲起,深入探讨反爬虫、多线程、分布式等进阶技术。以大量实例为支撑,覆盖网页、图片、音频等各类数据爬取ÿ…...
FreeRTOS(3)列表List
在 FreeRTOS 的源码中大量地使用了列表和列表项,因此想要深入学习 FreeRTOS,列表和列表项是必备的基础知识。这里所说的列表和列表项,是 FreeRTOS 源码中 List 和 List Item 的 直译,事实上, FreeRTOS 中的列表和列表项…...
C++和OpenGL实现3D游戏编程【连载23】——几何着色器和法线可视化
欢迎来到zhooyu的C++和OpenGL游戏专栏,专栏连载的所有精彩内容目录详见下边链接: 🔥C++和OpenGL实现3D游戏编程【总览】 1、本节实现的内容 上一节课,我们在Blend软件中导出经纬球模型时,遇到了经纬球法线导致我们在游戏中模型光照显示问题,我们在Blender软件中可以通过…...
Harmony开发笔记(未完成)
一、感想 作为一名拥有11年经验的Android开发者,我亲历了Android从高速发展到如今面临“僧多粥少”的过程。技术的世界瞬息万变,没有一种技术能够让人依赖一辈子。去年初,我自学了鸿蒙系统,并顺利通过了鸿蒙官方的初级和高级认。…...
【Java面试】创建线程有哪几种方式
目录 1.继承Thread类 2.实现Runnable接口 3.实现Callable接口和FutureTask 4.使用Executor框架(线程池) Java并发编程中不同接口和类之间的关系 总结 1.继承Thread类 优点: 简单直观。直接继承Thread类,可以方便地使用Threa…...
在Linux环境下利用MTCNN进行人脸检测(基于ncnn架构)
概述 本文将详细介绍如何在Linux环境下部署MTCNN模型进行人脸检测,并使用NCNN框架进行推理。 1. CMake的安装与配置 下载CMake源码 前往CMake官网下载,找到适合您系统的最新版本tar.gz文件链接,或者直接通过wget下载:CMake官方…...
