并发队列:ArrayBlockingQueue实际运用场景和原理
ArrayBlockingQueue实际应用场景
之前在某公司做过一款情绪识别的系统,这套系统通过调用摄像头接口采集人脸信息,将采集的人脸信息做人脸识别和情绪分析,最终经过一定的算法将个人情绪数据转化具体行为指标值。其中采集图片的部分就用到了并发队列ArrayBlockingQueue。

如上图所示:摄像头有n个,单线程采集的效率会比较慢,所以在采集摄像头的过程中是多线程的,另外采集到的图片需要存储到图片服务器,对图片服务器写也有很高的要求,图片服务器是集群的,也需要用到也多线程的。将图片入库后需要将图片数据打到人脸分析服务器上去处理,这部分涉及到了分布式消息,所以是黑色虚线部分用kafka来传递消息。其中红色虚线部分多线程图片采集将信息传递到多线程图片存储用到了ArrayBlockingQueue,它是并发安全队列。
ArrayBlockingQueue简化类图结构

从类图可以看出Queue接口提供了add,offer入队列的方法,提供poll出队列的方法!
BlockingQueue接口增加了put入队列的方法,提供take出队列的方法!
补充说明:UML类图结构:
- 继承:实线空箭头。
- 实现:虚线虚箭头。
并发队列阻塞和非阻塞概念
从上面类图名字可以看到Queue提供的方法是非阻塞的!而BlockingQueue提供的put,take方法是阻塞的!下面按老思路,我们用代码说明阻塞和非阻塞下!
非阻塞
import java.util.concurrent.ArrayBlockingQueue; /**
* @author :jiaolian
* @date :Created in 2021-02-02 20:16
* @description:ArrayBlockingQueue阻塞非阻塞测试
* @modified By:
* 公众号:叫练
*/
public class ArrayBlockingQueueTest {
public static void main(String[] args) {
ArrayBlockingQueue<String> arrayBlockingQueue = new ArrayBlockingQueue<>(1);
arrayBlockingQueue.offer("叫练");
arrayBlockingQueue.offer("叫练");
//输出arrayBlockingQueue的长度
System.out.println(arrayBlockingQueue.size());
}
}
如上代码:设置ArrayBlockingQueue长度为1,通过offer方法向队列添加2个元素,最后打印arrayBlockingQueue的长度?答案是1,不会阻塞,因为offer方法丢弃了第二个元素“叫练”,我们说出队和入队能够让其继续执行的队列我们称为非阻塞。如果换成add方法呢?就会报错队列溢出,如下图所示!但是还不是阻塞的。下面我们看看什么阻塞!

阻塞
import java.util.concurrent.ArrayBlockingQueue; /**
* @author :jiaolian
* @date :Created in 2021-02-02 20:16
* @description:ArrayBlockingQueue阻塞非阻塞测试
* @modified By:
* 公众号:叫练
*/
public class ArrayBlockingQueueTest {
public static void main(String[] args) throws InterruptedException {
ArrayBlockingQueue<String> arrayBlockingQueue = new ArrayBlockingQueue<>(1);
arrayBlockingQueue.put("叫练");
arrayBlockingQueue.put("叫练");
//输出arrayBlockingQueue的长度
System.out.println(arrayBlockingQueue.size());
}
}
如上代码:ArrayBlockingQueue长度为1,通过put方法向队列添加2个元素,最后输出arrayBlockingQueue的长度是多少?答案是控制台一直运行,因为在添加第二个“叫练”时程序阻塞了。我们说出队和入队不能够让其继续执行的队列我们称为阻塞,add方法,poll方法,take方法我们就不一一举例了,大家可以写代码做下最简单的测试!
好啦,我们对几个方法做个总结吧!
- 入队:
offer:队列满了丢弃。
add :队列满了报错。
put :阻塞。
- 出队:
poll :如果队列为空则返回null。
take :阻塞。
ArrayBlockingQueue实现原理浅析

如上图,ArrayBlockingQueue是用数组实现的,ReentrantLock独占锁控制数组的入队和出队。notEmpty,notFull是ReentrantLock的两个条件队列,用来控制队列是否进入阻塞状态,是生产者和消费者模型。下面我们看看take,put方法流程,其他的方法同理。
- take方法:多个线程竞争独占锁获取items[taskIndex]队首元素,其中A线程成功获取锁,其他线程阻塞等待A线程执行完成释放锁,如果队列不为空,A线程获取items[taskIndex]元素返回移除并释放锁让其他阻塞线程继续竞争;如果队列为空,A线程调用notEmpty.await方法进入条件队列并释放锁让其他阻塞线程继续竞争,其他线程发现队列为空也会进入notEmpty条件队列,等待put线程入队通知notEmpty阻塞线程。
- put方法:多个线程竞争独占锁设置items[putIndex]队尾元素,其中A线程成功获取锁,其他线程阻塞等待A线程执行完成释放锁,如果队列不满【队列长度】,A线程添加items[putIndex]元素返回并释放锁让其他阻塞线程继续竞争;如果队列满了,A线程调用notFull.await方法进入条件队列并释放锁让其他阻塞线程继续竞争,其他线程发现队列为空也会进入notFull条件队列,等待take线程出队通知notFull阻塞线程。
完全非阻塞队列ConcurrentLinkedQueue
ConcurrentLinkedQueue也实现了Queue接口,提供offer,add,poll方法都是非阻塞的,另外从名字可以看出,底层是链表结构,入队和出队用的是自旋的cas。
List 多线程安全方案:LinkedBlockingQueue

LinkedBlockingQueue和ArrayBlockingQueue 类似,LinkedBlockingQueue是有界的,长度是Integer.MAX_VALUE,实现上,LinkedBlockingQueue是链表,而且是双锁,如上图所示,takeLock独占锁控制队列头部,putLock控制队列尾部,互不影响,目的是提高LinkedBlockingQueue的并发度。
总结
今天我们介绍了并发队列重要的几个概念,整理出来希望能对你有帮助,写的比不全,同时还有许多需要修正的地方,希望亲们加以指正和点评,年前这段时间会继续输出线程池这些概念等。最后喜欢的请点赞加关注哦。点关注,不迷路,我是叫练【公众号】,边叫边练。

参考书籍:《Java并发编程之美》
并发队列:ArrayBlockingQueue实际运用场景和原理的更多相关文章
- JAVA并发(6)-并发队列ArrayBlockingQueue
本文讲ArrayBlockingQueue 1. 介绍 一个基于数组的有界阻塞队列,FIFO顺序.支持等待消费者和生产者线程的可选公平策略(默认是非公平的).公平的话通常会降低吞吐量,但是可以减少可变 ...
- 并发队列ConcurrentLinkedQueue、阻塞队列AraayBlockingQueue、阻塞队列LinkedBlockingQueue 区别和使用场景总结
三者区别与联系: 联系,三者 都是线程安全的.区别,就是 并发 和 阻塞,前者为并发队列,因为采用cas算法,所以能够高并发的处理:后2者采用锁机制,所以是阻塞的.注意点就是前者由于采用cas算 ...
- 自己总结 :并发队列ConcurrentLinkedQueue、阻塞队列AraayBlockingQueue、阻塞队列LinkedBlockingQueue 区别 和 使用场景总结
并发队列ConcurrentLinkedQueue.阻塞队列AraayBlockingQueue.阻塞队列LinkedBlockingQueue 区别 和 使用场景总结 分类: Java2013-0 ...
- 并发编程(八)—— Java 并发队列 BlockingQueue 实现之 ArrayBlockingQueue 源码分析
开篇先介绍下 BlockingQueue 这个接口的规则,后面再看其实现. 阻塞队列概要 阻塞队列与我们平常接触的普通队列(LinkedList或ArrayList等)的最大不同点,在于阻塞队列的阻塞 ...
- 并发队列之ArrayBlockingQueue
上一篇我们说了并发队列中的LinkedBlockingQueue队列,这次我们看看ArrayBlockingQueue,看看名字,我们想象一下LinkedList和ArrayList的区别,我们可以知 ...
- 并发编程-concurrent指南-阻塞队列-数组阻塞队列ArrayBlockingQueue
ArrayBlockingQueue类是实现了BlockingQueue. ArrayBlockingQueue是一个有界的阻塞队列,其内部实现是将对象放在一个数组中. 放入元素方法: (1) add ...
- 【Java并发】并发队列与线程池
并发队列 阻塞队列与非阻塞队 ConcurrentLinkedQueue BlockingQueue ArrayBlockingQueue LinkedBlockingQueue PriorityBl ...
- 解读 java 并发队列 BlockingQueue
点击添加图片描述(最多60个字)编辑 今天呢!灯塔君跟大家讲: 解读 java 并发队列 BlockingQueue 最近得空,想写篇文章好好说说 java 线程池问题,我相信很多人都一知半解的,包括 ...
- 10分钟搞定 Java 并发队列好吗?好的
| 好看请赞,养成习惯 你有一个思想,我有一个思想,我们交换后,一个人就有两个思想 If you can NOT explain it simply, you do NOT understand it ...
随机推荐
- 【译】深入理解Rust中的生命周期
原文标题:Understanding Rust Lifetimes 原文链接:https://medium.com/nearprotocol/understanding-rust-lifetimes- ...
- Interface注意事项
Interface 成员声明 声明属性,默认static & final 声明方法,默认public interface Instrument { int VALUE = 5; // stat ...
- Linux下使用acme.sh申请和管理Let’s Encrypt证书
关于Let's Encrypt 免费SSL证书 Let's Encrypt 作为一个公共且免费 SSL 的项目逐渐被广大用户传播和使用,是由 Mozilla.Cisco.Akamai.IdenTrus ...
- 2018年第九届蓝桥杯B组(201803-----乘积尾零)
标题题目:乘积尾零 如下的10行数据,每行有10个整数,请你求出它们的乘积的末尾有多少个零? 5650 4542 3554 473 946 4114 3871 9073 90 4329 2758 79 ...
- springboot源码解析-管中窥豹系列之排序(五)
一.前言 Springboot源码解析是一件大工程,逐行逐句的去研究代码,会很枯燥,也不容易坚持下去. 我们不追求大而全,而是试着每次去研究一个小知识点,最终聚沙成塔,这就是我们的springboot ...
- 【MyBatis】MyBatis 连接池和事务控制
MyBatis 连接池和事务控制 文章源码 MyBaits 连接池 实际开发中都会使用连接池,因为它可以减少获取连接所消耗的时间.具体可查看 MyBatis 数据源配置在 SqlMapConfig.x ...
- awk中的if ,else
local pct="$(awk -v one="$1" -v two="$2" 'BEGIN{ if (two > 0) { printf & ...
- 【Linux】ssh设置了密钥,但ssh登陆的时候还需要输入密码
------------------------------------------------------------------------------------------------- | ...
- SSRF - Pikachu
概述: SSRF(Server-Side Request Forgery:服务器端请求伪造) 其形成的原因大都是由于服务端提供了从其他服务器应用获取数据的功能,但又没有对目标地址做严格过滤与限制 导致 ...
- Ubuntu Terminal命令行新建仓库并推送到远程仓库
通常情况下,在本地新建一个仓库之后,需要在远端网页端也新建一个空的同名仓库,然后将两者进行关联才能推送. 那有没有办法直接在命令行就完成从新建到推送的过程而不需要中间在网页端也操作一番呢?办法当然是有 ...