抽象队列同步器AQS应用Lock

2024-01-03 00:48

本文主要是介绍抽象队列同步器AQS应用Lock,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  1. AQS特性

阻塞等待队列
共享/独占
公平/非公平
可重入
允许中断

  1. 锁的三大核心

自旋
lockSupport 来阻塞及唤醒
cas 算法 //比较与交换 unsafe 底层是依赖汇编指令 cmpxchg()
queue队列

AbstractQueuedSynchronizer
exclusiveOwnerThread 当前获取锁的线程是谁
aqs依赖一个整形变量state 为0 锁没被持有
队列是基于 内部类Node 双向链表 同步等待队列

可重入锁是state反复加1

  1. 多个线程竞争锁(公平锁)

获取当前线程
获取state 状态 ,如state为0 且 阻塞队列中没有其他线程则获取锁
exclusiveOwnerThread 指向当前线程
如state不为0,则判断当前持有锁的线程是不是自己,是自己则加1,如果不是自己持有的继续CAS。返回false,加锁失败。
加锁失败的线程直接加入队列

protected final boolean tryAcquire(int acquires) {final Thread current = Thread.currentThread();int c = getState();if (c == 0) {if (!hasQueuedPredecessors() &&compareAndSetState(0, acquires)) {setExclusiveOwnerThread(current);return true;}}else if (current == getExclusiveOwnerThread()) {int nextc = c + acquires;if (nextc < 0)throw new Error("Maximum lock count exceeded");setState(nextc);return true;}return false;
}
  1. tryAcquire(arg) 锁竞争逻辑

acquireQueued(addWaiter(Node.EXCLUSIVE), arg) 线程入队逻辑
队列中的Node 除了有共享,和独占之外的属性
waitState(整型) 标识当前节点的生命状态:信号量
sinnal =-1 可被唤醒
cancelled =1 出现异常 中断引起 需要被废弃
condition =-2 条件等待
propagate = -3 广播状态 传播
0 初始状态

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

当线程入队时,位于head节点的线程会重新再抢一次锁
如果还没抢到锁,head节点才会把自己阻塞(此时第一轮循环把head节点置为sinnal=-1 代表可被唤醒。第二轮循环阻塞线程,并且判断线程是否有中断信号唤醒的),如果抢到锁则会直接出队列

注: head节点thread为null

final boolean acquireQueued(final Node node, int arg) {boolean failed = true;try {boolean interrupted = false;for (;;) {final Node p = node.predecessor();if (p == head && tryAcquire(arg)) {setHead(node);p.next = null; // help GCfailed = false;return interrupted;}if (shouldParkAfterFailedAcquire(p, node) &&parkAndCheckInterrupt())interrupted = true;}} finally {if (failed)cancelAcquire(node);}
}

shouldParkAfterFailedAcquire(p, node)
waitState = 0 ->-1 在持有锁的线程T0在释放锁之后,要判断head节点的waitState 是否!=0 ,如果waitState!=0 成立会再次waitState = 0 ->-1 。要想唤醒排队的第一个线程T1,T1被唤醒准备继续走循环拿锁。在非公平的情况下可能还会再次失败,此时可能T3拿到了锁!而T1会被再次阻塞,head节点会再次经历两次循环 waitState = 0 ->-1

Park阻塞线程唤醒的两种方式
中断
release()

线程被中断唤醒:抛异常
cancelAcquire(node);//标记当前节点为canceld,信号量,我们要剔除这样的节点

  1. 阻塞队列 BlockingQueue

线程通信的一个工具,在任意时刻、不管并发有多高,在单jvm上,同一时间永远只有一个线程能对队列进行出队或者入队操作
线程安全的队列

以ArrayBlockingQueue 为例

public ArrayBlockingQueue(int capacity, boolean fair) {if (capacity <= 0)throw new IllegalArgumentException();this.items = new Object[capacity];lock = new ReentrantLock(fair); //创建锁notEmpty = lock.newCondition(); //条件对象notFull =  lock.newCondition(); //条件对象
}public void put(E e) throws InterruptedException {checkNotNull(e);final ReentrantLock lock = this.lock;lock.lockInterruptibly();     //生产者加锁try {while (count == items.length) //队列满了notFull.await();     enqueue(e);} finally {lock.unlock();}
}

//await可打开, 一进方法内,
1、先判断线程是否被中断,中断则抛出异常。
2、不被中断则会入Node队列(条件等待队列)。
3、通过锁状态释放锁。
4、通知CLH队列(竞争锁时线程的队列:同步队列)消费下一个节点。
注:线程获取的条件:只有在CLH队列里等待的node节点并且该节点的前一个结点是sinal,条件队列里的线程是不可以获取锁的。
注2:当有一个线程从queue队列中消费,留出一个空时。会发送一个sinal消息给条件队列这时候条件队列会转换成CLH队列

这篇关于抽象队列同步器AQS应用Lock的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/564118

相关文章

如何通过Python实现一个消息队列

《如何通过Python实现一个消息队列》这篇文章主要为大家详细介绍了如何通过Python实现一个简单的消息队列,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录如何通过 python 实现消息队列如何把 http 请求放在队列中执行1. 使用 queue.Queue 和 reque

解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)

《解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)》该文章介绍了使用Redis的阻塞队列和Stream流的消息队列来优化秒杀系统的方案,通过将秒杀流程拆分为两条流水线,使用Redi... 目录Redis秒杀优化方案(阻塞队列+Stream流的消息队列)什么是消息队列?消费者组的工作方式每

5分钟获取deepseek api并搭建简易问答应用

《5分钟获取deepseekapi并搭建简易问答应用》本文主要介绍了5分钟获取deepseekapi并搭建简易问答应用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需... 目录1、获取api2、获取base_url和chat_model3、配置模型参数方法一:终端中临时将加

JavaScript中的isTrusted属性及其应用场景详解

《JavaScript中的isTrusted属性及其应用场景详解》在现代Web开发中,JavaScript是构建交互式应用的核心语言,随着前端技术的不断发展,开发者需要处理越来越多的复杂场景,例如事件... 目录引言一、问题背景二、isTrusted 属性的来源与作用1. isTrusted 的定义2. 为

Python调用另一个py文件并传递参数常见的方法及其应用场景

《Python调用另一个py文件并传递参数常见的方法及其应用场景》:本文主要介绍在Python中调用另一个py文件并传递参数的几种常见方法,包括使用import语句、exec函数、subproce... 目录前言1. 使用import语句1.1 基本用法1.2 导入特定函数1.3 处理文件路径2. 使用ex

Redis延迟队列的实现示例

《Redis延迟队列的实现示例》Redis延迟队列是一种使用Redis实现的消息队列,本文主要介绍了Redis延迟队列的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录一、什么是 Redis 延迟队列二、实现原理三、Java 代码示例四、注意事项五、使用 Redi

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链