解锁Java并发编程的秘密武器!揭秘AQS,让你的代码从此告别‘锁’事烦恼,多线程同步不再是梦!

简介: 【8月更文挑战第25天】AbstractQueuedSynchronizer(AQS)是Java并发包中的核心组件,作为多种同步工具类(如ReentrantLock和CountDownLatch等)的基础。AQS通过维护一个表示同步状态的`state`变量和一个FIFO线程等待队列,提供了一种高效灵活的同步机制。它支持独占式和共享式两种资源访问模式。内部使用CLH锁队列管理等待线程,当线程尝试获取已持有的锁时,会被放入队列并阻塞,直至锁被释放。AQS的巧妙设计极大地丰富了Java并发编程的能力。

在Java的并发编程中,AbstractQueuedSynchronizer(简称AQS)是一个核心组件,它不仅是实现同步器的基础,也是并发包中多种锁(如ReentrantLock、CountDownLatch等)的底层实现。AQS通过其精巧的设计,为开发者提供了一种高效且灵活的同步机制。

AQS的核心概念
AQS是一个抽象类,全称为AbstractQueuedSynchronizer,它定义了一种基于FIFO(先进先出)队列的同步框架。AQS内部维护了一个volatile的state变量,用于表示同步状态。这个状态变量是AQS的核心,通过它来控制对共享资源的访问。当state为0时,表示没有线程持有锁;当state大于0时,表示有线程持有锁。

AQS支持两种资源共享模式:独占式和共享式。独占式模式下,每次只有一个线程能够持有锁,如ReentrantLock;而共享式模式下,允许多个线程同时访问共享资源,如ReentrantReadWriteLock的读锁部分。

AQS的内部结构
AQS内部使用了一个CLH(Craig, Landin, and Hagersten)队列来管理等待获取锁的线程。这个队列是一个双向链表,通过head和tail两个指针来维护队列的头部和尾部。每个节点(Node)代表一个等待获取锁的线程,节点中包含了线程引用、等待状态等信息。

AQS的工作原理
当一个线程尝试获取锁时,首先会检查state的值。如果state为0,表示当前没有线程持有锁,该线程将成功获取锁,并将state设置为1(或其他值,取决于具体实现)。如果state不为0,表示锁已被其他线程持有,当前线程将被放入等待队列中,并进入阻塞状态。

当持有锁的线程释放锁时,它会将state的值设置为0,并唤醒等待队列中的下一个线程。被唤醒的线程会再次尝试获取锁,如果成功,则继续执行;如果失败,则重新进入等待队列。

示例代码
下面是一个使用AQS实现简单互斥锁的示例代码:

java
import java.util.concurrent.locks.AbstractQueuedSynchronizer;

class Mutex {
private final Sync sync = new Sync();

public void lock() {  
    sync.acquire(1);  
}  

public void unlock() {  
    sync.release(1);  
}  

private static class Sync extends AbstractQueuedSynchronizer {  
    @Override  
    protected boolean tryAcquire(int acquires) {  
        return compareAndSetState(0, 1);  
    }  

    @Override  
    protected boolean tryRelease(int releases) {  
        setState(0);  
        return true;  
    }  

    @Override  
    protected boolean isHeldExclusively() {  
        return getState() == 1;  
    }  
}  

}

// 使用示例
public class Main {
public static void main(String[] args) {
Mutex mutex = new Mutex();

    // 线程1尝试获取锁  
    new Thread(() -> {  
        mutex.lock();  
        try {  
            // 模拟任务执行  
            Thread.sleep(1000);  
        } catch (InterruptedException e) {  
            e.printStackTrace();  
        } finally {  
            mutex.unlock();  
        }  
    }).start();  

    // 线程2尝试获取锁(将在线程1释放锁后获取)  
    new Thread(() -> {  
        mutex.lock();  
        try {  
            // 模拟任务执行  
            Thread.sleep(1000);  
        } catch (InterruptedException e) {  
            e.printStackTrace();  
        } finally {  
            mutex.unlock();  
        }  
    }).start();  
}  

}
在这个示例中,我们定义了一个名为Mutex的互斥锁类,它内部使用了一个继承自AbstractQueuedSynchronizer的Sync类来实现锁的逻辑。通过重写tryAcquire、tryRelease和isHeldExclusively方法,我们实现了简单的锁获取和释放逻辑。

AQS以其简洁而强大的设计,为Java并发编程提供了坚实的基础。通过理解AQS的工作原理,我们可以更加深入地掌握Java并发编程的精髓。

相关文章
|
1月前
|
设计模式 人工智能 安全
AQS:Java 中悲观锁的底层实现机制
AQS(AbstractQueuedSynchronizer)是Java并发包中实现同步组件的基础工具,支持锁(如ReentrantLock、ReadWriteLock)和线程同步工具类(如CountDownLatch、Semaphore)等。Doug Lea设计AQS旨在抽象基础同步操作,简化同步组件构建。 使用AQS需实现`tryAcquire(int arg)`和`tryRelease(int arg)`方法以获取和释放资源,共享模式还需实现`tryAcquireShared(int arg)`和`tryReleaseShared(int arg)`。
99 32
AQS:Java 中悲观锁的底层实现机制
|
24天前
|
算法 Java 调度
Java多线程基础
本文主要讲解多线程相关知识,分为两部分。第一部分涵盖多线程概念(并发与并行、进程与线程)、Java程序运行原理(JVM启动多线程特性)、实现多线程的两种方式(继承Thread类与实现Runnable接口)及其区别。第二部分涉及线程同步(同步锁的应用场景与代码示例)及线程间通信(wait()与notify()方法的使用)。通过多个Demo代码实例,深入浅出地解析多线程的核心知识点,帮助读者掌握其实现与应用技巧。
|
2月前
|
Java
【源码】【Java并发】【AQS】从ReentrantLock、Semaphore、CutDownLunch、CyclicBarrier看AQS源码
前言 主播觉得,AQS的原理,就是通过这2个队列的协助,实现核心功能,同步队列(CLH队列)和条件队列(Condition队列)。 同步队列(CLH队列) 作用:管理需要获...
89 18
【源码】【Java并发】【AQS】从ReentrantLock、Semaphore、CutDownLunch、CyclicBarrier看AQS源码
|
1月前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
|
1月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
79 0
|
3月前
|
设计模式 存储 安全
【Java并发】【AQS】适合初学者体质的AQS入门
AQS这是灰常重要的哈,很多JUC下的框架的核心,那都是我们的AQS,所以这里,我们直接开始先研究AQS。 那说到研究AQS,那我们应该,使用开始说起🤓 入门 什么是AQS? AQS(Abst
98 8
【Java并发】【AQS】适合初学者体质的AQS入门
|
3月前
|
Java 数据库
【YashanDB知识库】kettle同步大表提示java内存溢出
在数据导入导出场景中,使用Kettle进行大表数据同步时出现“ERROR:could not create the java virtual machine!”问题,原因为Java内存溢出。解决方法包括:1) 编辑Spoon.bat增大JVM堆内存至2GB;2) 优化Kettle转换流程,如调整批量大小、精简步骤;3) 合理设置并行线程数(PARALLELISM参数)。此问题影响所有版本,需根据实际需求调整相关参数以避免内存不足。
|
3月前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
146 23
|
4月前
|
安全 Java 开发者
Java并发迷宫:同步的魔法与死锁的诅咒
在Java并发编程中,合理使用同步机制可以确保线程安全,避免数据不一致的问题。然而,必须警惕死锁的出现,采取适当的预防措施。通过理解同步的原理和死锁的成因,并应用有效的设计和编码实践,可以构建出高效、健壮的多线程应用程序。
94 21
|
2月前
|
数据采集 存储 网络协议
Java HttpClient 多线程爬虫优化方案
Java HttpClient 多线程爬虫优化方案

热门文章

最新文章