`
臻是二哥
  • 浏览: 189490 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
博客专栏
Group-logo
Java技术分享
浏览量:0
社区版块
存档分类
最新评论

JUC之AQS

阅读更多
AQS是同步框架,它进行两个方面的工作:资源的管理和资源申请者的管理。对应由两部分组成:一个volatile int state(代表共享资源)和一个FIFO线程等待队列(多线程争用资源被阻塞时会进入此队列)。state的访问方式有三种:
	getState()
	setState()
	compareAndSetState()


AQS定义两种资源共享方式:Exclusive(独占,只有一个线程能执行,如ReentrantLock)和Share(共享,多个线程可同时执行,如Semaphore/CountDownLatch)。

实现自定义同步器时只需要实现共享资源state的获取与释放方式即可,至于具体线程等待队列的维护(如获取资源失败入队/唤醒出队等),AQS已经在顶层实现好了。

自定义同步器实现时主要实现以下几种方法:
	isHeldExclusively():该线程是否正在独占资源。只有用到condition才需要去实现它。
	tryAcquire(int):独占方式。尝试获取资源,成功则返回true,失败则返回false。
	tryRelease(int):独占方式。尝试释放资源,成功则返回true,失败则返回false。
	tryAcquireShared(int):共享方式。尝试获取资源。负数表示失败;0表示成功,但没有剩余可用资源;正数表示成功,且有剩余资源。
	tryReleaseShared(int):共享方式。尝试释放资源,成功则返回true,失败则返回false。


以ReentrantLock为例,state初始化为0,表示未锁定状态。A线程lock()时,会调用tryAcquire()独占该锁并将state+1。此后,其他线程再tryAcquire()时就会失败,直到A线程unlock()到state=0(即释放锁)为止,其它线程才有机会获取该锁。当然,释放锁之前,A线程自己是可以重复获取此锁的(state会累加),这就是可重入的概念。但要注意,获取多少次就要释放多么次,这样才能保证state是能回到零态的。

  再以CountDownLatch以例,任务分为N个子线程去执行,state也初始化为N(注意N要与线程个数一致)。这N个子线程是并行执行的,每个子线程执行完后countDown()一次,state会CAS减1。等到所有子线程都执行完后(即state=0),会unpark()主调用线程,然后主调用线程就会从await()函数返回,继续后余动作。

一般来说,自定义同步器要么是独占方法,要么是共享方式,他们也只需实现tryAcquire-tryRelease、tryAcquireShared-tryReleaseShared中的一种即可。但AQS也支持自定义同步器同时实现独占和共享两种方式,如ReentrantReadWriteLock。
理解了这些,可以很容易的自定义同步器类,如下例子:

class Mutex implements Lock , Serializable{
    private static class Syn extends AbstractQueuedSynchronizer{
        @Override
        protected boolean isHeldExclusively() {
            return getState()==1;
        }
        @Override
        protected boolean tryAcquire(int arg) {
            assert arg==1;
            if(compareAndSetState(0,1)) {
                setExclusiveOwnerThread(Thread.currentThread());
                return true;
            }
            return false;
        }
        @Override
        protected boolean tryRelease(int arg) {
            assert arg==1;
            if(getState()==0)
                throw new IllegalMonitorStateException();
            setExclusiveOwnerThread(null);
            setState(0);
            return true;
        }
    }
    private Syn syn=new Syn();
    public void lock() {
        syn.acquire(1);
    }
    public void unlock() {
        syn.release(1);
    }
    public void lockInterruptibly() throws InterruptedException{
        syn.acquireInterruptibly(1);
    }
    public boolean tryLock(){
        return syn.tryAcquire(1);
    }
    public boolean tryLock(long time, TimeUnit unit) throws InterruptedException{
        return syn.tryAcquireNanos(1,unit.toNanos(time));
    }
    public Condition newCondition(){
        return null;
    }
}


上面说,AQS对资源申请者的管理已经在顶层实现好了,自定义同步器时仅仅需要重写几种特定的方法,不需要关心队列的实现细节。但这里还是简要的了解下。
先看acquire()方法:acquire()方法是独占模式获取资源的顶级入口。代码如下:

 public final void acquire(int arg) {
     if (!tryAcquire(arg) &&
         acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
         selfInterrupt();
 }


流程如下:
1. 调用自定义同步器的tryAcquire()尝试直接去获取资源,如果成功则直接返回;
2. 没成功,则addWaiter()将该线程加入等待队列的尾部,并标记为独占模式;
3. acquireQueued()使线程在等待队列中休息,有机会时(轮到自己,会被unpark())会去尝试获取资源。获取到资源后才返回。如果在整个等待过程中被中断过,则返回true,否则返回false。
4. 如果线程在等待过程中被中断过,它是不响应的。只是获取资源后才再进行自我中断selfInterrupt(),将中断补上。

再看release()方法,release()方法是独占模式下释放资源的顶级入口。代码如下:

public final boolean release(int arg) {
    if (tryRelease(arg)) {
        Node h = head;//找到头结点
        if (h != null && h.waitStatus != 0)
            unparkSuccessor(h);//唤醒等待队列里的下一个线程
        return true;
    }
    return false;
}

至此,关于AQS的简单介绍就结束了。AQS是许多常用类的基础,比如ReentrantLock,Semaphore,ReentrantReadWriteLock,CountDownLatch等等,都是基于AQS构建的。感兴趣的话可以去看他们的实现。
0
1
分享到:
评论

相关推荐

    juc aqs java

    juc 的aqs介绍。

    java并发编程:juc、aqs

    `AQS`(AbstractQueuedSynchronizer)是JUC库中的一个关键组件,它是一个抽象基类,为构建自定义的同步器提供了基础框架。AQS通过内部维护一个基于链表的等待队列,有效地管理线程的同步和唤醒,从而实现锁和其他同步...

    JUC AQS的加解锁.pdf

    Java的并发编程是多线程和多任务处理的核心技术之一,而在Java并发包 java.util.concurrent 中,AQS(AbstractQueuedSynchronizer)扮演了至关重要的角色。AQS是一种框架,用来构建锁或其他同步组件的基础。它提供了...

    JUC(一)-AQS源码分析

    为了学习JUC,AQS是基础中的基础,所以我们首先深入了解下AQS。 一、锁的介绍 为了了解AQS的源码,我们需要先大概下锁中的一些功能 1.1 乐观锁/悲观锁 乐观锁与悲观锁是一种广义上的概念,体现了看待线程同步的不同...

    JUC核心类AQS的底层原理

    AQS作为Java并发工具包(JUC)中的一个核心抽象类,其设计目的是为了实现各种同步器(如锁、信号量等)。AQS主要通过三个核心组成部分来实现这些同步组件的功能: 1. **State变量及其CAS操作**:AQS维护了一个名为`...

    JUC AQS(AbstractQueuedSynchronizer)

    ReentrantLock Lock 加锁过程源码分析图,AQS 源码分析

    AQS和JUC知识点讲解

    《AQS和JUC知识点详解》 在Java并发编程领域,AbstractQueuedSynchronizer(AQS)和Java Util Concurrency(JUC)是两个至关重要的概念。它们为开发高效、线程安全的多线程程序提供了强大的工具。本文将深入解析这...

    Java volatile与AQS锁内存可见性

    从JUC中的AQS引入,讲解Java volatile与AQS锁内存可见性

    java8源码-java-tutorial:Java基础实战教程,包含Java并发包JUC,AQS实战,设计模式实战,基础面试实战,IO/NI

    java8 源码 Java基础实战教程 模块 说明 dive-in-concurrent 并发相关实战教程 dive-in-design-pattern 设计模式相关实战教程 dive-in-interview Java面试相关 dive-in-io ...Geekerstar-个人主页:

    Java 多线程与并发(10-26)-JUC锁- 锁核心类AQS详解.pdf

    Java并发包(java.util.concurrent,简称JUC)提供了一系列工具和类库来帮助开发者简化并发编程的工作。其中,AbstractQueuedSynchronizer(简称AQS)是构建各种同步器的核心组件。 AQS是一个抽象的队列同步器,它...

    java并发编程-AQS和JUC实战

    ### Java并发编程-AQS和JUC实战 #### 一、ReentrantLock 重入锁 **1.1 概述** - **基本介绍**: `ReentrantLock` 是一个实现了 `Lock` 接口的可重入互斥锁,提供比 `synchronized` 更丰富的功能。与 `synchronized...

    JUC并发工具包实例.zip

    Java并发工具包(Java Concurrency Utility,简称JUC)是Java平台中用于高效并发编程的重要模块,它在`java.util.concurrent`包下提供了一系列高级并发工具。这些工具可以帮助开发者更好地管理和控制多线程环境,...

    精心整理的AQS和JUC相关的面试题.pdf【ReentrantLock】

    ⼀、ReentrantLock重⼊锁 ...1.5> AQS源码解析 ⼆、Condition重⼊锁的搭配类 三、Semaphore信号量 四、ReadWriteLock读写锁 五、CountDownLatch倒计时器 六、CyclicBarrier循环栅栏 七、LockSupport线程阻塞⼯具类

    05-Java多线程并发编程JUC.pdf

    - AQS(AbstractQueuedSynchronizer):提供了一种实现阻塞锁和相关同步器的框架。 以上知识点涵盖了Java多线程并发编程和JUC的基础与高级特性,从线程的状态、创建、启动、通信、同步机制,到线程池的使用和线程...

    aqs中文翻译-by sushengmiyan.pdf

    java大师doug lean 在JDK1.5版本的AQS论文中文翻译。 许可:本作品的全部或部分在不为牟利或商业利益为目的的,且在第一页引述本声明及全完整引用的前提下,以数码或硬拷贝形式供个人或课堂使用的复制或分发不收取...

    java编发编程:JUC综合讲解

    4. **同步器(Synchronizers)**:JUC库中的同步器主要是通过AQS(AbstractQueuedSynchronizer)来实现的。AQS是一个抽象类,为锁和同步器提供了一种通用的底层实现。它维护了一个等待队列,提供了基于FIFO的等待...

    并发锁核心类AQS学习笔记

    JUC 包中的同步类基本都是基于 AQS 同步器来实现的,如 ReentrantLock,Semaphore 等。 二、原理 1、AQS 工作机制: 如果被请求的共享资源空闲,则将当前请求资源的线程设置为有效的工作线程,并且将共享资源设置为...

    Java-JUC-多线程 进阶

    8 锁的现象是 Java 中的一种并发编程模型,描述了 8 种不同的锁机制,它们是:公平锁、非公平锁、可重入锁、读写锁、StampedLock、LockSupport、AbstractQueuedSynchronizer(AQS)、ReentrantLock。这些锁机制可以...

    Java并发编程与高并发解决方案.txt

    AbstractQueuedSynchronizer(AQS)是J.U.C中的一个核心框架,它为实现依赖于“first-in-first-out”(FIFO)等待队列的阻塞锁和相关同步器提供了一个框架。`ReentrantLock`、`Semaphore`、`CountDownLatch`等都是...

    Chinese_AQS:添加中文注释, 重新按业务部门开发工程师的思维角度组织代码中的方法顺序, 使中国程序员轻松理解 AbstractQueuedSynchronizer;

    AQS是Java并发库(Java Concurrency Utility,JUC)的一部分,由Doug Lea设计,是Java语言中实现锁和同步器的一种高效、灵活的框架。它抽象出一个基于队列的同步器模型,内部维护了一个状态(state)以及一个等待...

Global site tag (gtag.js) - Google Analytics