并发队列: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 ...
随机推荐
- Vs2017编译器提示:不能将“const char *”类型的值分配到“char *”类型的实体
在项目属性中将语言符合模式改成否即可
- linux 笔记的注意事项
声明:本人Linux的笔记是根据<鸟哥私房菜>而写的 command [-option] parameter1 parameter2 ... command 是命令的名称: [ ]中括号是 ...
- mysql词法分析和语法分析
如果没有命中查询缓存,就要开始真正执行语句了.首先,MySQL 需要知道你要做什么,因此需要对 SQL 语句做解析.分析器先会做"词法分析".你输入的是由多个字符串和空格组成的一条 ...
- 2021年了,`IEnumerator`、`IEnumerable`还傻傻分不清楚?
IEnumerator.IEnumerable这两个接口单词相近.含义相关,傻傻分不清楚. 入行多年,一直没有系统性梳理这对李逵李鬼. 最近本人在怼着why神的<其实吧,LRU也就那么回事> ...
- .NET Core引入日志(Log4Net篇)
Demo版本信息如下: VS:2019 框架:.Net Core 3.1 Log4Net:2.0.12 思维导图: [1]添加依赖项 通过nuget添加Log4Net [2]创建公共类 添加公共类Lo ...
- Linux学习笔记 | 配置nginx
目录 一.Nginx概述 二.why Nginx? 三.Linux安装Nginx APT源安装 官网源码安装 四.nginx相关文件的配置 html文件:/var/www/html/index.htm ...
- Memcached、Redis、Mongodb比较
Memcached(内存Cache) Memcached 是一个高性能的分布式内存对象缓存系统.通过在内存里维护一个统一的巨大的hash表,它能够用来存储各种格式的数据,包括图像.视频.文件以及数据库 ...
- LeetCode993. 二叉树的堂兄弟节点
题目 1 class Solution { 2 public: 3 TreeNode* r1;TreeNode* r2; 4 bool isCousins(TreeNode* root, int x, ...
- nodejs内网穿透
说明 本地服务注册,基于子域名->端口映射.公网测试请开启二级或三级域名泛解析 无心跳保活.无多线程并发处理 服务器端 请求ID基于全局变量,不支持PM2多进程开服务端.(多开请修改uid函数, ...
- 【工具篇】Mysql的安装和使用
[导读]Mysql是数据分析师入门级的技能之一,对于很多小白同学来说,可能还没有机会接触SQL知识.那么我们如何熟悉和练习SQL呢,今天教大家安装两个软件:MySQL和Navicat.后续我们会推出S ...