Java 并发之 AQS

1. AQS

AQS(AbstractQueuedSynchronizer) 是一个用来构建锁和同步器的框架,内部定义了很多锁的相关方法,使用 AQS 能简单高效的构造出大量的同步器,常见的 ReentrantLockReentrantReadWriteLockCountDownLatchSemaphoreSychronousQueue 等都是基于 AQS 的。

1.1 AQS 原理

AQS 内部维护了一个 volatile int state 来表示同步状态,通过内置的 FIFO 队列来完成获取资源线程的排队工作。AQS 使用 CAS 对改同步状态进行原子操作实现对其值的修改。

protected final int getState() {
	return state;
}

protected final void setState(int newState) {
	state = newState;
}

protected final boolean compareAndSetState(int expect, int update) {
	return unsafe.compareAndSwapInt(this, stateOffset, expect, update);
}

1.2 AQS 对资源的共享方式

  • Exclusive 独占,比如 ReentrantLock
  • Share 共享,多个线程共享,比如 SemaphoreCountDownLatch

1.3 模板模式

  • 同步器的设计是基于模板设计模式的,一般自定义同步器的方式:
    • 继承 AQS,重写其指定方法
    • 将 AQS 组合在自定义同步组件的实现中,调用其模板方法,这些模板方法会优先调用重写的方法
  • 需要重写的 AQS 方法:
    • isHeldExclusively() 改线程是否正在独占资源
    • tryAcquire(int) 尝试独占获取资源
    • tryRelease(int) 尝试独占释放资源
    • tryAcquireShared(int) 共享方式获取资源
    • tryReleaseShared(int) 共享方式释放资源
  • 这些方法在 AQS 里都是直接 throw new UnsupportedOperationException();
  • 一般来说,同步器只需要实现独占的方法或者共享方法中的一种就够了,但是也支持两种方式,比如 ReentrantReadWriteLock

2. 同步器

2.1 Semaphore 信号量

  • 允许多个线程同时访问
  • 执行 acquire 阻塞,直到有一个许可证可以获得然后拿走一个许可证,也可以多个
  • release 方法是释放一个许可,可能会释放一个 acquire 阻塞的线程,也可以多个
  • 主要是维持了一个可获得许可证的数量,常用于限制某资源的线程数量
  • Semaphore 的构造方法也可以设置公平和非公平模式

2.2 CountDownLatch 倒计时器

  • 允许 count 个线程阻塞在一个地方,知道所有的线程执行完毕
  • countDown 方法其实就是调用 tryReleaseShared 以 CAS 的方式减少 state,当 state 为 0 时表示所有线程都执行力 countDown,所有阻塞的线程会继续执行下去
  • state 不为 0 的时候,把线程放入阻塞队列,自旋检查 state
  • state 设为 n,适用于主线程等待其他线程然后一起执行
  • state 设为 1,其他都 countDownLatch.await() 住,一执行 countDown 多个线程同时执行,可以实现线程的并行性
  • CountDownLatch 是一次性的,只能再初始化的时候设置 state,没有其它方法能改变这个值

2.3 CyclicBarrier 循环栅栏

  • CountDownLatch 很类似,但是是基于 ReetrantLockCondition
  • 就是让一组线程到达一个屏障的时候被阻塞,直到最后一个线程到达这个屏障就释放所有 await 的线程
  • 构造函数中可以添加 Runnable,实现类似于回调函数的功能,当所有线程到达屏障时,执行方法
  • 可以实现多线程计算,再统计的功能
  • CountDownLatch 重点是一个线程等待,其它线程在执行完某件事之后可以终止,也可以等待
  • CyclicBarrier 重点是所有线程到达,阀门打开继续执行,一个线程没有完成其他都要等待

2.4 ReetrantLock

  • ReentrantReadWriteLock 相比于 ReetrantLock 可以保证多线程同时读
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
AQS(AbstractQueuedSynchronizer)是Java并发编程中的一个重要类,它可以理解为抽象的队列同步器。AQS提供了一种基于FIFO队列的同步机制,用于实现各种同步器,如ReentrantLock、CountDownLatch、Semaphore等。 AQS的核心思想是使用一个volatile的int类型变量state来表示同步状态,通过CAS(Compare and Swap)操作来实现对state的原子更新。AQS内部维护了一个双向链表,用于保存等待获取同步状态的线程。 AQS的具体实现包括以下几个方面: 1. 内部属性:AQS内部有两个重要的属性,一个是head,表示队列的头节点;另一个是tail,表示队列的尾节点。 2. 入队操作:AQS的入队操作是通过enq方法实现的。在入队操作中,首先判断队列是否为空,如果为空,则需要初始化队列;否则,将新节点添加到队列的尾部,并更新tail指针。 3. CAS操作:AQS的CAS操作是通过compareAndSetHead和compareAndSetTail方法实现的。这些方法使用CAS操作来更新head和tail指针,保证操作的原子性。 4. 出队操作:AQS的出队操作是通过deq方法实现的。在出队操作中,首先判断队列是否为空,如果为空,则返回null;否则,将头节点出队,并更新head指针。 5. 同步状态的获取和释放:AQS提供了acquire和release方法来获取和释放同步状态。acquire方法用于获取同步状态,如果获取失败,则会将当前线程加入到等待队列中;release方法用于释放同步状态,并唤醒等待队列中的线程。 通过继承AQS类,可以实现自定义的同步器。具体的实现方式是重写AQS的几个关键方法,如tryAcquire、tryRelease等,来实现对同步状态的获取和释放。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值