当前位置: 首页 > news >正文

【JUC并发编程】ArrayBlockingQueue和LinkedBlockingQueue源码2分钟看完

文章目录

  • 1、BlockingQueue
    • 1)接口方法
    • 2)阻塞队列分类
  • 2、ArrayBlockingQueue
    • 1)构造函数
    • 2)put()入队
    • 3)take()出队
  • 3、LinkedBlockingQueue
    • 1)构造函数
    • 2)put()入队
    • 3)take()出队

1、BlockingQueue

BlockingQueue是JUC包下提供的一个阻塞队列 接口;

1)接口方法

在这里插入图片描述

队列操作

  • 抛出异常:add(e)、remove()、element()
  • 返回特定值:offer()队尾入队/poll()删除队头元素/peek()
  • 一直阻塞:put(e)/take()
  • 超时退出:offer(e,time,unit)/poll(time,unit)

其中:BlockingQueue 不接受 null 元素。试图 add 、 put 或 offer ⼀个 null 元素时,某些实现会抛出 NullPointerException 。

在这里插入图片描述

2)阻塞队列分类

  • ArrayBlockingQueue:由数组结构组成的有界阻塞队列。

  • LinkedBlockingQueue:由链表组成的有界阻塞队列。

  • PriorityBlockingQueue:一个支持优先级排序的无界阻塞队列(默认升序排序)。

  • DelayQueue:支持延时获取元素的无界阻塞队列。

    队列使用PriorityQueue来实现。

  • SynchronousQueue:一个不存储元素的阻塞队列。

  • LinkedTransferQueue:由链表组成的无界阻塞队列。

    其多了tryTransfer()方法和transfer()方法。

    • transfer():可以把生产者传入的元素立刻传输给消费者。如果没有consumer在等待接收元素,transfer方法会将元素存放在队列的tail节点,并等到该元素被消费者消费了才返回。
    • tryTransfer():用来试探生产者传入的元素是否能直接传给消费者。不管是否有consumer正在等待接收元素,都立刻返回。

2、ArrayBlockingQueue

ArrayBlockingQueue是由数组结构组成的有界阻塞队列。通过ReentrantLock保证线程安全、并实现 Producer-Consumer模式。

1)构造函数

在这里插入图片描述

从构造函数可知,默认采用非公平锁;

public ArrayBlockingQueue(int capacity, boolean fair) {if (capacity <= 0)throw new IllegalArgumentException();// 底层使用对象数组保存元素this.items = new Object[capacity];// 初始化需要加锁使用的ReentrantLock实例,默认采用非公平锁lock = new ReentrantLock(fair);/*** 判断队列是 空 or 满*     notEmpty⽤于执⾏take时进⾏await()等待操作,put时进⾏signal()唤醒操作*     notFull⽤于执⾏take时进⾏signal()唤醒操作,put时进⾏await()等待操作*/notEmpty = lock.newCondition();notFull =  lock.newCondition();
}

方法释义:

  • 构造函数的入参capacity指定了底层存储元素数组⻓度的⼤⼩;
  • 初始化需要加锁使⽤的ReentrantLock实例,默认采⽤的是⾮公平锁;
  • 基于Lock的Condition判断队列是 空 or 满
    • notEmpty⽤于执⾏take时进⾏await()等待操作,put时进⾏signal()唤醒操作;
    • notFull⽤于执⾏take时进⾏signal()唤醒操作,put时进⾏await()等待操作;

2)put()入队

public void put(E e) throws InterruptedException {// 入参不能为空checkNotNull(e);final ReentrantLock lock = this.lock;// 加可中断锁lock.lockInterruptibly();try {while (count == items.length)// 队列满了,则阻塞等待signal唤醒,同时释放次有的锁。notFull.await();// 入队操作enqueue(e);} finally {lock.unlock();}
}private void enqueue(E x) {final Object[] items = this.items;items[putIndex] = x;// 如果数据已经插入到数组末尾,则重置putIndex为0,从0开始继续插入。if (++putIndex == items.length)putIndex = 0;count++;// 通知take线程解除阻塞notEmpty.signal();
}

方法释义:

  • ⾸先尝试获得可中断锁,即:lock.lockInterruptibly(),当执⾏interrupt操作时,该锁可以被中断。
  • 如果数组中元素的个数(count)等于数组的⻓度了,说明队列已经满了,在该线程上执⾏等待操作:notFull.await(); ,等待signal唤醒。
  • 如果队列没有满,则调⽤enqueue(e)⽅法执⾏⼊列操作;
    1. ⼊列操作⾸先会将待插⼊值x放⼊数组下标为putIndex的位置上,然后再将putIndex加1,来指向下⼀次插⼊的下标位置。
      • 如果加1后的putIndex等于了数组的⻓度,那么说明已经越界了(因为putIndex是从0开始的);做循环式插⼊,重置putIndex为0,从0开始继续插入。
  • 最后,执⾏count++来计算元素总个数;并调⽤notEmpty.signal()⽅法来解除阻塞;
    • 当队列为空的时候,执⾏take⽅法会被notEmpty.await()阻塞;
    • 此处就是通知take线程解除阻塞

3)take()出队

public E take() throws InterruptedException {final ReentrantLock lock = this.lock;lock.lockInterruptibly();try {while (count == 0)// 如果队列为空,则线程阻塞 等待signal唤醒,释放持有的锁notEmpty.await();// 执行出队操作return dequeue();} finally {lock.unlock();}
}private E dequeue() {final Object[] items = this.items;@SuppressWarnings("unchecked")// 获取takeIndex下标的元素E x = (E) items[takeIndex];// 将takeIndex下标下的袁术置为null,便于后面GC回收items[takeIndex] = null;// 如果出队操作的到了数组末尾,则重置takeIndex,从0开始继续取出if (++takeIndex == items.length)takeIndex = 0;count--;// 默认itrs为null,不会走进if代码段。if (itrs != null)itrs.elementDequeued();// 通知put线程解除阻塞notFull.signal();return x;
}

方法释义:

  • take⽅法和put⽅法类似,区别是出队的指针是takeIndex;

  • ⾸先尝试获得可中断锁,即:lock.lockInterruptibly(),当执⾏interrupt操作时,该锁可以被中断。

  • 如果队列中为空;执⾏notEmpty.await()将线程阻塞 等待signal唤醒,释放持有的锁。

    • 当调⽤put⽅法向队列中放⼊元素之后 ,会调⽤notEmpty.signal⽅法对等待的线程执⾏唤醒操作;
  • 如果出队操作的到了数组末尾,则重置takeIndex,从0开始继续取出;

  • 出队执⾏完毕后,调⽤notFull.signal⽅法来唤醒在notFull上⾯

    await的线程。 通知put线程解除阻塞。

3、LinkedBlockingQueue

LinkedBlockingQueue是由链表结构组成的有界阻塞队列。通过ReentrantLock保证线程安全、并实现 Producer-Consumer模式。

1)构造函数

如果不指定容量,则默认LinkedBlockingQueue是无界阻塞队列(capacity = Integer.MAX_VALUE)

在这里插入图片描述

构造函数中会创建⼀个空的节点,作为整个链表的头节点。

2)put()入队

在这里插入图片描述

private void enqueue(Node<E> node) {// assert putLock.isHeldByCurrentThread();// assert last.next == null;last = last.next = node;
}

整个put()流程和ArrayBlockingQueue基本一致,对于链表容量的统计会额外采用一个AtomiceInteger类型的变量count维护。

    • 最后唤醒put()线程的代码段上有一个 c == 0的判断,这里的c是入队操作之前的数量。
/** Current number of elements */
private final AtomicInteger count = new AtomicInteger();/** Lock held by take, poll, etc */
private final ReentrantLock takeLock = new ReentrantLock();/** Wait queue for waiting takes */
private final Condition notEmpty = takeLock.newCondition();/** Lock held by put, offer, etc */
private final ReentrantLock putLock = new ReentrantLock();/** Wait queue for waiting puts */
private final Condition notFull = putLock.newCondition();

这里有个比较有意思的点:

  • 入队操作添加完元素之后,如果发现当前队列的元素数量还没到最大容量,则尝试唤醒其他put()操作阻塞的线程;
if (c + 1 < capacity)notFull.signal();

3)take()出队

在这里插入图片描述

private E dequeue() {// assert takeLock.isHeldByCurrentThread();// assert head.item == null;Node<E> h = head;Node<E> first = h.next;h.next = h; // help GChead = first;E x = first.item;first.item = null;return x;
}

整个take()流程和ArrayBlockingQueue基本一致,稍微看一下即可。

  • 最后唤醒take()线程的代码段上有一个 c == capacity的判断,这里的c是出队操作之前的数量。

和LinkedBlockingQueue的put()操作一样:

  • 出队操作移除完元素之后,如果发现当前队列的元素数量 > 1,则尝试唤醒其他take()操作阻塞的线程;
if (c > 1)notEmpty.signal();

相关文章:

【JUC并发编程】ArrayBlockingQueue和LinkedBlockingQueue源码2分钟看完

文章目录1、BlockingQueue1&#xff09;接口方法2&#xff09;阻塞队列分类2、ArrayBlockingQueue1&#xff09;构造函数2&#xff09;put()入队3&#xff09;take()出队3、LinkedBlockingQueue1&#xff09;构造函数2&#xff09;put()入队3&#xff09;take()出队1、Blocking…...

GitHub个人资料自述与管理主题设置

目录 关于您的个人资料自述文件 先决条件 添加个人资料自述文件 删除个人资料自述文件 管理主题设置 补充&#xff1a;建立一个空白文件夹 关于您的个人资料自述文件 可以通过创建个人资料 README&#xff0c;在 GitHub.com 上与社区分享有关你自己的信息。 GitHub 在个…...

Express篇-连接mysql

创建数据库配置文件config/sqlconfig.jsconst sqlconfig {host: localhost, // 连接地址user: root, //用户名password: ****, //密码port: 3306 , //端口号database: mysql01_dbbooks //数据库名 } module.exports sqlconfig封装数据库管理工具 utils/mysqlUtils.…...

win10 安装rabbitMQ详细步骤

win10 安装rabbitMQ详细步骤 win10 安装rabbitMQ详细步骤win10 安装rabbitMQ详细步骤一、下载安装程序二、安装配置erlang三、安装rabbitMQ四、验证初始可以通过用户名&#xff1a;guest 密码guest来登录。报错&#xff1a;安装RabbitMQ出现Plugin configuration unchanged.问题…...

【成为架构师课程系列】一线架构师:6个经典困惑及其解法

目录 一线架构师:6个经典困惑及其解法 多阶段还是多视图&#xff1f; 内置最佳实践 架构方法论:3个阶段&#xff0c;一个贯穿 Pre-architecture阶段&#xff1a;ADMEMS矩阵方法 Conceptual Architecture阶段&#xff1a;重大需求塑造做概念架构 Refined Architecture…...

光耦合器的定义与概述

光耦合器或光电耦合器是一种电子元件&#xff0c;基本上充当具有不同电压电平的两个独立电路之间的接口。光耦合器是可在输入和输出源之间提供电气隔离的常用元件。它是一个 6 引脚器件&#xff0c;可以有任意数量的光电探测器。 在这里&#xff0c;光源发出的光束作为输入和输…...

谷粒商城--品牌管理详情

目录 1.简单上传测试 2.Aliyun Spring Boot OSS 3.模块mall-third-service 4.前端 5.数据校验 6.JSR303数据校验 7.分组校验功能 8.自定义校验功能 9.完善代码 1.简单上传测试 OSS是对象存储服务&#xff0c;有什么用呢&#xff1f;把图片存储到云服务器上能让所有人…...

stack、queue和priority_queue

目录 一、栈&#xff08;stack&#xff09; 1.stack的使用 2.容器适配器 3.stack的模拟实现 二、队列&#xff08;queue&#xff09; 1.queue的使用 2.queue的模拟实现 三、双端队列&#xff08;deque&#xff09; 1.vector&#xff0c;list的优缺点 2.认识deque 四…...

面试题(二十二)消息队列与搜索引擎

2. 消息队列 2.1 MQ有什么用&#xff1f; 参考答案 消息队列有很多使用场景&#xff0c;比较常见的有3个&#xff1a;解耦、异步、削峰。 解耦&#xff1a;传统的软件开发模式&#xff0c;各个模块之间相互调用&#xff0c;数据共享&#xff0c;每个模块都要时刻关注其他模…...

Spring Security in Action 第三章 SpringSecurity管理用户

本专栏将从基础开始&#xff0c;循序渐进&#xff0c;以实战为线索&#xff0c;逐步深入SpringSecurity相关知识相关知识&#xff0c;打造完整的SpringSecurity学习步骤&#xff0c;提升工程化编码能力和思维能力&#xff0c;写出高质量代码。希望大家都能够从中有所收获&#…...

Java面试——maven篇

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…...

基于微信小程序的游戏账号交易小程序

文末联系获取源码 开发语言&#xff1a;Java 框架&#xff1a;ssm JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7/8.0 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;eclipse/myeclipse/idea Maven包&#xff1a;Maven3.3.9 浏览器…...

Matlab绘制隐函数总结-二维和三维

1.二维隐函数 二维隐函数满足f(x,y)0f(x,y)0f(x,y)0&#xff0c;这里无法得到yf(x)yf(x)yf(x)的形式。不能通过普通函数绘制。 我们要关注的是使用fplot函数和fimplicit函数。 第1种情况&#xff1a;基本隐函数 基本的隐函数形式形如&#xff1a; x2y22x2(x2y2)12x^{2}y^{…...

如何直观地理解傅立叶变换?频域和时域的理解

如何直观地理解傅立叶变换 傅里叶变换连续形式的傅立叶变换如何直观地理解傅立叶变换?一、傅里叶级数1.1傅里叶级数的三角形式1.2 傅里叶级数的复指数形式二、傅里叶变换2.1一维连续傅里叶变换三、频谱和功率谱3.1频谱的获得3.2频谱图的特征3.3频谱图的组成频域(frequency do…...

STC15读取内部ID示例程序

STC15读取内部ID示例程序&#x1f389;本案例基于STC15F2K60S2为验证对象。 &#x1f4d1;STC15 ID序列介绍 STC15系列STC最新一代STC15系列单片机出厂时都具有全球唯一身份证号码(ID号)。最新STC15系列单片机的程序存储器的最后7个字节单元的值是全球唯一ID号&#xff0c;用…...

Xml格式化与高亮显示

具体请参考&#xff1a;Xml格式化与高亮显示...

【GlobalMapper精品教程】045:空间分析工具(2)——相交

GlobalMapper提供的空间分析(操作)的方法有:交集、并集、单并集、差异、对称差集、相交、重叠、接触、包含、等于、内部、分离等,本文主要讲述相交工具的使用。 文章目录 一、实验数据二、符号化设置三、相交运算四、结果展示五、心灵感悟一、实验数据 加载配套实验数据(…...

4年外包终上岸,我只能说这类公司能不去就不去..

我大学学的是计算机专业&#xff0c;毕业的时候&#xff0c;对于找工作比较迷茫&#xff0c;也不知道当时怎么想的&#xff0c;一头就扎进了一家外包公司&#xff0c;一干就是4年。现在终于跳槽到了互联网公司了&#xff0c;我想说的是&#xff0c;但凡有点机会&#xff0c;千万…...

sklearn降维算法1 - 降维思想与PCA实现

目录1、概述1.1 维度概念2、PCA与SVD2.1 降维实现2.2 重要参数n_components2.2.1 案例&#xff1a;高维数据的可视化2.2.2 最大似然估计自选超参数2.2.3 按信息量占比选超参数1、概述 1.1 维度概念 shape返回的结果&#xff0c;几维几个方括号嵌套 特征矩阵特指二维的 一般来…...

「期末复习」线性代数

第一章 行列式 行列式是一个数&#xff0c;是一个结果三阶行列式的计算&#xff1a;主对角线的乘积全排列与对换逆序数为奇就为奇排列&#xff0c;逆序数为偶就为偶排列对换&#xff1a;定理一&#xff1a;一个排列的任意两个元素对换&#xff0c;排列改变奇偶性&#xff08;和…...

【BBF系列协议】TR369 客户端实现的功能点

CPE制造商(包括现代的、路由器、智能家居集线器、IPTV STB、ATA/VoIP、存储设备、媒体中心、Femtocell、IP电话和摄像头)预计将允许通过TR-069和TR-369(用户服务平台)标准对这些设备进行远程管理。这些标准允许服务提供商,包括ISP、电信运营商等,简化这些CPE设备的配置、…...

突破AI推理瓶颈:Ivy混合精度技术如何平衡速度与精度

突破AI推理瓶颈&#xff1a;Ivy混合精度技术如何平衡速度与精度 【免费下载链接】ivy unifyai/ivy: 是一个基于 Python 的人工智能库&#xff0c;支持多种人工智能算法和工具。该项目提供了一个简单易用的人工智能库&#xff0c;可以方便地实现各种人工智能算法的训练和推理&am…...

Fish Speech 1.5新手教程:Gradio界面布局解读、滑块参数含义与推荐值

Fish Speech 1.5新手教程&#xff1a;Gradio界面布局解读、滑块参数含义与推荐值 1. 认识Fish Speech 1.5的Gradio界面 Fish Speech 1.5提供了一个直观的Web界面&#xff0c;让即使没有编程经验的用户也能轻松使用语音合成功能。整个界面采用左右分栏设计&#xff0c;左侧是输…...

FireRed-OCR Studio快速上手:使用Gradio替代Streamlit构建更轻量Web界面

FireRed-OCR Studio快速上手&#xff1a;使用Gradio替代Streamlit构建更轻量Web界面 1. 为什么选择Gradio重构FireRed-OCR Studio&#xff1f; 如果你用过FireRed-OCR Studio&#xff0c;一定会被它强大的文档解析能力所吸引。它能精准识别文字、还原复杂表格、提取数学公式&…...

HOG特征可视化:不用深度学习也能看懂图像特征(OpenCV+Matplotlib教程)

HOG特征可视化&#xff1a;不用深度学习也能看懂图像特征&#xff08;OpenCVMatplotlib教程&#xff09; 当你第一次看到"HOG特征"这个词时&#xff0c;可能会联想到猪的鼻子或是某种动物特征。但实际上&#xff0c;HOG&#xff08;Histogram of Oriented Gradients&…...

构建企业级人工智能高质量数据集:方法与路径

姜春宇 白玉真 刘渊 王超伦&#xff08;中国信息通信研究院&#xff0c;北京 100191&#xff09;摘 要 当前&#xff0c;我国人工智能数据集面临质量评估方法缺失、能力建设体系不明确等挑战。梳理了人工智能数据集的构成和分类&#xff0c;结合结构化数据质量评估&#xff0c;…...

StructBERT扩展应用:小说情感脉络分析工具开发

StructBERT扩展应用&#xff1a;小说情感脉络分析工具开发 1. 引言 你有没有读过一本小说后&#xff0c;感觉整个故事的情感起伏特别精彩&#xff0c;但却说不清楚具体是怎么变化的&#xff1f;或者作为文学研究者&#xff0c;想要量化分析一部作品的情感发展脉络&#xff1f…...

.NET 8 打造工业级运动控制系统

前言工业自动化与智能制造快速发展&#xff0c;高精度、高响应的运动控制系统已成为设备核心。然而&#xff0c;传统运动控制开发往往面临接口复杂、文档缺失、调试困难等挑战&#xff0c;严重制约了研发效率与系统稳定性。本文推荐一款专为正运动 ZMotion 系列运动控制器设计的…...

目前可靠的硅胶干燥剂源头厂家排行榜

硅胶干燥剂源头厂家排行榜&#xff1a;专业深度测评开篇&#xff1a;定下基调随着科技的发展和生活品质的提高&#xff0c;硅胶干燥剂因其高效、环保的特性&#xff0c;已成为防潮、防霉的重要产品。本次测评旨在为消费者提供一份可靠的硅胶干燥剂源头厂家排行榜&#xff0c;帮…...

三机九节点电力系统 Simulink 仿真模型探索

【三机九节点电力系统Simulink仿真模型】 3机9节点Matlab/Simulink电力系统仿真模型 1个风机 2个同步机 风电渗透率20.7%最近在研究电力系统仿真&#xff0c;搭建了一个超有意思的三机九节点 Matlab/Simulink 电力系统仿真模型&#xff0c;来和大家分享一下。这个模型可不简单…...