解锁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并发编程的精髓。

相关文章
|
6月前
|
Java 开发工具
【Azure Storage Account】Java Code访问Storage Account File Share的上传和下载代码示例
本文介绍如何使用Java通过azure-storage-file-share SDK实现Azure文件共享的上传下载。包含依赖引入、客户端创建及完整示例代码,助你快速集成Azure File Share功能。
463 6
|
7月前
|
IDE Java 关系型数据库
Java 初学者学习路线(含代码示例)
本教程为Java初学者设计,涵盖基础语法、面向对象、集合、异常处理、文件操作、多线程、JDBC、Servlet及MyBatis等内容,每阶段配核心代码示例,强调动手实践,助你循序渐进掌握Java编程。
875 3
|
7月前
|
安全 Java 应用服务中间件
Spring Boot + Java 21:内存减少 60%,启动速度提高 30% — 零代码
通过调整三个JVM和Spring Boot配置开关,无需重写代码即可显著优化Java应用性能:内存减少60%,启动速度提升30%。适用于所有在JVM上运行API的生产团队,低成本实现高效能。
821 3
|
6月前
|
Java 数据处理 API
为什么你的Java代码应该多用Stream?从循环到声明式的思维转变
为什么你的Java代码应该多用Stream?从循环到声明式的思维转变
333 115
|
6月前
|
安全 Java 编译器
为什么你的Java代码需要泛型?类型安全的艺术
为什么你的Java代码需要泛型?类型安全的艺术
239 98
|
7月前
|
Java
java入门代码示例
本文介绍Java入门基础,包含Hello World、变量类型、条件判断、循环及方法定义等核心语法示例,帮助初学者快速掌握Java编程基本结构与逻辑。
547 0
|
6月前
|
安全 Java 容器
告别繁琐判空:Optional让你的Java代码更优雅
告别繁琐判空:Optional让你的Java代码更优雅
|
6月前
|
安全 Java 容器
告别空指针噩梦:Optional让Java代码更优雅
告别空指针噩梦:Optional让Java代码更优雅
482 94
|
6月前
|
Java 编译器 API
java最新版和java8的区别,用代码展示
java最新版和java8的区别,用代码展示
504 43