博客
关于我
【腾讯阿里最全面试题】介绍下Synchronized、Volatile、CAS、AQS,以及各自的使用场景...
阅读量:275 次
发布时间:2019-03-01

本文共 1256 字,大约阅读时间需要 4 分钟。

AQS(AbstractQueuedSynchronizer)是Java并发工具包中一个核心的同步框架,广泛应用于实现多种锁和同步组件,如ReentrantLock、CountDownLatch等。以下是对AQS的详细分析:

AQS的结构与机制

AQS维护了一个状态值和一个FIFO双向队列,用于管理线程的同步和等待。状态值通过volatile修饰确保线程间的可见性,保证锁的状态一致性。队列中的节点包含线程和相关状态信息,用于处理线程的等待和唤醒。

核心组件

  • 状态值(state):用于表示锁的状态,可能用于计数重入次数或其他同步逻辑。
  • 双向队列:FIFO结构,用于存储等待锁的线程,确保先进先出的顺序。
  • 队列节点

    每个节点包含以下信息:

    • waitStatus:线程的等待状态,可能为SIGNAL、CONDITION等值。
    • 前驱节点(prev)后驱节点(next),构成FIFO队列。
    • 当前线程(thread):等待锁的线程。

    AQS的工作流程

  • 获取锁:调用lock()方法,尝试修改状态值和队列。如果成功,线程继续执行;如果失败,线程被加入队列等待。
  • 释放锁:调用unlock()方法,修改状态值,并从队列中取出线程,唤醒它。
  • 节点插入:线程竞争锁失败时,构造节点并插入队列。
  • 节点删除:线程获取锁后,删除自己从队列中的节点。
  • 唤醒线程:线程释放锁后,唤醒队列中的下一个线程。
  • AQS的锁类型

    AQS支持两种资源共享方式:

  • 独占锁:同一时间仅有一个线程持有锁,适用于ReentrantLock。
  • 共享锁:多个线程可同时持有锁,适用于ReentrantReadWriteLock。
  • ReentrantLock的实现

    ReentrantLock基于AQS实现独占锁。它的lock()方法通过递增状态值获取锁,unlock()方法递减状态值释放锁。线程在竞争锁时,使用循环的acquire()方法,通过CAS操作尝试获取锁,失败时加入队列。

    CountDownLatch的实现

    CountDownLatch基于AQS的独占锁实现。它维护一个计数器,countDown()方法递减计数器,当计数器为零时,等待的线程被唤醒。

    性能与优化

    AQS使用自旋机制,线程在释放锁后继续执行,减少等待时间。Java6后引入偏向锁优化,允许线程在无竞争情况下直接获取锁,进一步提升性能。

    应用场景

    • ReentrantLock:用于需要重入锁控制的场景,比如资源池管理。
    • ReentrantReadWriteLock:用于读写锁控制,支持多个读线程同时访问,但一旦有写线程进入,所有读线程等待。
    • CountDownLatch:用于等待多个线程完成任务后,唤醒主线程。
    • Semaphore:用于控制资源的可用次数,允许多个线程同时使用资源。

    总结

    AQS通过状态值和FIFO队列实现线程同步,支持独占和共享锁模式。其灵活性和高效性使其成为Java并发工具包的核心同步框架。理解AQS有助于深入掌握Java多线程编程中的锁与同步机制。

    转载地址:http://aaqa.baihongyu.com/

    你可能感兴趣的文章
    MySQL 常用列类型
    查看>>
    mysql 常用命令
    查看>>
    Mysql 常见ALTER TABLE操作
    查看>>
    MySQL 常见的 9 种优化方法
    查看>>
    MySQL 常见的开放性问题
    查看>>
    Mysql 常见错误
    查看>>
    mysql 常见问题
    查看>>
    MYSQL 幻读(Phantom Problem)不可重复读
    查看>>
    mysql 往字段后面加字符串
    查看>>
    mysql 快速自增假数据, 新增假数据,mysql自增假数据
    查看>>
    Mysql 批量修改四种方式效率对比(一)
    查看>>
    Mysql 报错 Field 'id' doesn't have a default value
    查看>>
    MySQL 报错:Duplicate entry 'xxx' for key 'UNIQ_XXXX'
    查看>>
    Mysql 拼接多个字段作为查询条件查询方法
    查看>>
    mysql 排序id_mysql如何按特定id排序
    查看>>
    Mysql 提示:Communication link failure
    查看>>
    mysql 插入是否成功_PDO mysql:如何知道插入是否成功
    查看>>
    Mysql 数据库InnoDB存储引擎中主要组件的刷新清理条件:脏页、RedoLog重做日志、Insert Buffer或ChangeBuffer、Undo Log
    查看>>
    mysql 数据库中 count(*),count(1),count(列名)区别和效率问题
    查看>>
    mysql 数据库备份及ibdata1的瘦身
    查看>>