简介

java中Collection集合有三大家族List,Set和Queue。当然Map也算是一种集合类,但Map并不继承Collection接口。

List,Set在我们的工作中会经常使用,通常用来存储结果数据,而Queue由于它的特殊性,通常用在生产者消费者模式中。

现在很火的消息中间件比如:Rabbit MQ等都是Queue这种数据结构的展开。

今天这篇文章将带大家进入Queue家族。

Queue接口

先看下Queue的继承关系和其中定义的方法:

Queue继承自Collection,Collection继承自Iterable。

Queue有三类主要的方法,我们用个表格来看一下他们的区别:

方法类型方法名称方法名称区别Insertaddoffer两个方法都表示向Queue中添加某个元素,不同之处在于添加失败的情况,add只会返回true,如果添加失败,会抛出异常。offer在添加失败的时候会返回false。所以对那些有固定长度的Queue,优先使用offer方法。Removeremovepoll如果Queue是空的情况下,remove会抛出异常,而poll会返回null。Examineelementpeek获取Queue头部的元素,但不从Queue中删除。两者的区别还是在于Queue为空的情况下,element会抛出异常,而peek返回null。

注意,因为对poll和peek来说null是有特殊含义的,所以一般来说Queue中禁止插入null,但是在实现中还是有一些类允许插入null比如LinkedList。尽管如此,我们在使用中还是要避免插入null元素。

Queue的分类

一般来说Queue可以分为BlockingQueue,Deque和TransferQueue三种。

BlockingQueue

BlockingQueue是Queue的一种实现,它提供了两种额外的功能:

  1. 当当前Queue是空的时候,从BlockingQueue中获取元素的操作会被阻塞。
  2. 当当前Queue达到最大容量的时候,插入BlockingQueue的操作会被阻塞。

BlockingQueue的操作可以分为下面四类:

操作类型Throws exceptionSpecial valueBlocksTimes outInsertadd(e)offer(e)put(e)offer(e, time, unit)Removeremove()poll()take()poll(time, unit)Examineelement()peek()not applicablenot applicable

第一类是会抛出异常的操作,当遇到插入失败,队列为空的时候抛出异常。

第二类是不会抛出异常的操作。

第三类是会Block的操作。当Queue为空或者达到最大容量的时候。

第四类是time out的操作,在给定的时间里会Block,超时会直接返回。

BlockingQueue是线程安全的Queue,可以在生产者消费者模式的多线程中使用,如下所示:

class Producer implements Runnable {private final BlockingQueue queue;Producer(BlockingQueue q) { queue = q; }public void run() {try {while (true) { queue.put(produce()); }} catch (InterruptedException ex) { ... handle ...}}Object produce() { ... }}class Consumer implements Runnable {private final BlockingQueue queue;Consumer(BlockingQueue q) { queue = q; }public void run() {try {while (true) { consume(queue.take()); }} catch (InterruptedException ex) { ... handle ...}}void consume(Object x) { ... }}class Setup {void main() {BlockingQueue q = new SomeQueueImplementation();Producer p = new Producer(q);Consumer c1 = new Consumer(q);Consumer c2 = new Consumer(q);new Thread(p).start();new Thread(c1).start();new Thread(c2).start();}}

最后,在一个线程中向BlockQueue中插入元素之前的操作happens-before另外一个线程中从BlockQueue中删除或者获取的操作。

Deque

Deque是Queue的子类,它代表double ended queue,也就是说可以从Queue的头部或者尾部插入和删除元素。

同样的,我们也可以将Deque的方法用下面的表格来表示,Deque的方法可以分为对头部的操作和对尾部的操作:

方法类型Throws exceptionSpecial valueThrows exceptionSpecial valueInsertaddFirst(e)offerFirst(e)addLast(e)offerLast(e)RemoveremoveFirst()pollFirst()removeLast()pollLast()ExaminegetFirst()peekFirst()getLast()peekLast()

和Queue的方法描述基本一致,这里就不多讲了。

当Deque以 FIFO (First-In-First-Out)的方法处理元素的时候,Deque就相当于一个Queue。

当Deque以LIFO (Last-In-First-Out)的方式处理元素的时候,Deque就相当于一个Stack。

TransferQueue

TransferQueue继承自BlockingQueue,为什么叫Transfer呢?因为TransferQueue提供了一个transfer的方法,生产者可以调用这个transfer方法,从而等待消费者调用take或者poll方法从Queue中拿取数据。

还提供了非阻塞和timeout版本的tryTransfer方法以供使用。

我们举个TransferQueue实现的生产者消费者的问题。

先定义一个生产者:

@Slf4j
@Data
@AllArgsConstructor
class Producer implements Runnable {private TransferQueue<String> transferQueue;private String name;private Integer messageCount;public static final AtomicInteger messageProduced = new AtomicInteger();@Overridepublic void run() {for (int i = 0; i < messageCount; i++) {try {boolean added = transferQueue.tryTransfer( "第"+i+"个", 2000, TimeUnit.MILLISECONDS);log.info("transfered {} 是否成功: {}","第"+i+"个",added);if(added){messageProduced.incrementAndGet();}} catch (InterruptedException e) {log.error(e.getMessage(),e);}}log.info("total transfered {}",messageProduced.get());}
}

在生产者的run方法中,我们调用了tryTransfer方法,等待2秒钟,如果没成功则直接返回。

再定义一个消费者:

@Slf4j
@Data
@AllArgsConstructor
public class Consumer implements Runnable {private TransferQueue<String> transferQueue;private String name;private int messageCount;public static final AtomicInteger messageConsumed = new AtomicInteger();@Overridepublic void run() {for (int i = 0; i < messageCount; i++) {try {String element = transferQueue.take();log.info("take {}",element );messageConsumed.incrementAndGet();Thread.sleep(500);} catch (InterruptedException e) {log.error(e.getMessage(),e);}}log.info("total consumed {}",messageConsumed.get());}}

在run方法中,调用了transferQueue.take方法来取消息。

下面先看一下一个生产者,零个消费者的情况:

@Testpublic void testOneProduceZeroConsumer() throws InterruptedException {TransferQueue<String> transferQueue = new LinkedTransferQueue<>();ExecutorService exService = Executors.newFixedThreadPool(10);Producer producer = new Producer(transferQueue, "ProducerOne", 5);exService.execute(producer);exService.awaitTermination(50000, TimeUnit.MILLISECONDS);exService.shutdown();}

输出结果:

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第0个 是否成功: false
[pool-1-thread-1] INFO com.flydean.Producer - transfered 第1个 是否成功: false
[pool-1-thread-1] INFO com.flydean.Producer - transfered 第2个 是否成功: false
[pool-1-thread-1] INFO com.flydean.Producer - transfered 第3个 是否成功: false
[pool-1-thread-1] INFO com.flydean.Producer - transfered 第4个 是否成功: false
[pool-1-thread-1] INFO com.flydean.Producer - total transfered 0

可以看到,因为没有消费者,所以消息并没有发送成功。

再看下一个有消费者的情况:

@Testpublic void testOneProduceOneConsumer() throws InterruptedException {TransferQueue<String> transferQueue = new LinkedTransferQueue<>();ExecutorService exService = Executors.newFixedThreadPool(10);Producer producer = new Producer(transferQueue, "ProducerOne", 2);Consumer consumer = new Consumer(transferQueue, "ConsumerOne", 2);exService.execute(producer);exService.execute(consumer);exService.awaitTermination(50000, TimeUnit.MILLISECONDS);exService.shutdown();}

输出结果:

[pool-1-thread-2] INFO com.flydean.Consumer - take 第0个
[pool-1-thread-1] INFO com.flydean.Producer - transfered 第0个 是否成功: true
[pool-1-thread-2] INFO com.flydean.Consumer - take 第1个
[pool-1-thread-1] INFO com.flydean.Producer - transfered 第1个 是否成功: true
[pool-1-thread-1] INFO com.flydean.Producer - total transfered 2
[pool-1-thread-2] INFO com.flydean.Consumer - total consumed 2

可以看到Producer和Consumer是一个一个来生产和消费的。

总结

本文介绍了Queue接口和它的三大分类,这三大分类又有非常多的实现类,我们将会在后面的文章中再详细介绍。

欢迎关注我的公众号:程序那些事,更多精彩等着您!更多内容请访问

一文弄懂java中的Queue家族​www.flydean.com

deque stack java_一文弄懂java中的Queue家族相关推荐

  1. queue double java_一文弄懂java中的Queue家族

    java中Queue家族简介 简介 java中Collection集合有三大家族List,Set和Queue.当然Map也算是一种集合类,但Map并不继承Collection接口. List,Set在 ...

  2. 一文弄懂java中的Queue家族

    文章目录 简介 Queue接口 Queue的分类 BlockingQueue Deque TransferQueue 总结 java中Queue家族简介 简介 java中Collection集合有三大 ...

  3. 一文弄懂Java中线程池原理

    在工作中,我们经常使用线程池,但是你真的了解线程池的原理吗?同时,线程池工作原理和底层实现原理也是面试经常问的考题,所以,今天我们一起聊聊线程池的原理吧. 为什么要用线程池 使用线程池主要有以下三个原 ...

  4. 一文弄懂神经网络中的反向传播法

    最近在看深度学习的东西,一开始看的吴恩达的UFLDL教程,有中文版就直接看了,后来发现有些地方总是不是很明确,又去看英文版,然后又找了些资料看,才发现,中文版的译者在翻译的时候会对省略的公式推导过程进 ...

  5. java中date类型如何赋值_一文读懂java中的Reference和引用类型

    简介 java中有值类型也有引用类型,引用类型一般是针对于java中对象来说的,今天介绍一下java中的引用类型.java为引用类型专门定义了一个类叫做Reference.Reference是跟jav ...

  6. 一文弄懂神经网络中的反向传播法——BackPropagation【转】

    本文转载自:https://www.cnblogs.com/charlotte77/p/5629865.html 一文弄懂神经网络中的反向传播法--BackPropagation 最近在看深度学习的东 ...

  7. 一文读懂Java中File类、字节流、字符流、转换流

    一文读懂Java中File类.字节流.字符流.转换流 第一章 递归:File类: 1.1:概述 java.io.File 类是文件和目录路径名的抽象表示,主要用于文件和目录的创建.查找和删除等操作. ...

  8. 一文弄懂Java线程安全队列

    文章目录 一.分类 二.BlockingQueue 阻塞队列 三.ConcurrentLinkedQueue 非阻塞队列 一.分类 java中所有队列都继承至java.util.Queue接口,该接口 ...

  9. 一文简单弄懂tensorflow_【TensorFlow】一文弄懂CNN中的padding参数

    在深度学习的图像识别领域中,我们经常使用卷积神经网络CNN来对图像进行特征提取,当我们使用TensorFlow搭建自己的CNN时,一般会使用TensorFlow中的卷积函数和池化函数来对图像进行卷积和 ...

最新文章

  1. 精通JavaScript--07设计模式:行为型
  2. 20172303 2017-2018-2 《程序设计与数据结构》第4周学习总结
  3. C++新旧类型转换小记
  4. web基础html元素制作web
  5. 传智播客JavaWeb day11--事务的概念、事务的ACID、数据库锁机制、
  6. 数据库-mysql基础操作之输入查询
  7. python天天向上每十天休息一天_Python基础第十天
  8. java单引号转义_Java基础入门——Java语言基础(上)
  9. 深入理解CPU cache:组织、一致性(同步)、编程
  10. Javascript中的null、undefined、NaN
  11. hua图软件 mac_细数Mac上那些好用且免费的软件(四)
  12. cocos2dx lua 打印和保存日志
  13. psp c语言编程软件,PSP2000自制系统3.03OE-C如何傻瓜安装及系统测试
  14. 用算法去扫雷(go语言)
  15. A股各概念板块龙头股大全
  16. 逆天了!全地形、四舵轮、八连杆、独立悬挂的机器人运动结构方案,来了!
  17. golang 解析 --- 进程,线程,协程
  18. 循环-05. 兔子繁衍问题
  19. 虹科新闻|ATTO 宣布支持 Apple 最新操作系统 macOS® 13 Ventura
  20. 最全Linux面试题

热门文章

  1. “三行代码,确实需要耗上一整天”
  2. K8s稳居容器榜首,Docker冲顶技术热词,微服务应用热度不减,2021云原生开发者现状
  3. 2 年增长 1 万亿!继苹果之后,微软市值也突破 2 万亿美元
  4. 会Python了不起吗?是的,简直开挂!(文末有福利)
  5. 众善之源 or 万恶之源?详析微服务的好与坏
  6. 翻车事故频发,原来是开发者漏了这一步!
  7. 共建智慧云基石,阿里云携手英特尔走向数智未来
  8. 优秀的程序员真的不写注释吗? | 原力计划
  9. 阿里云科学家入选计算机顶会 HPCA 名人堂,他是什么来头?
  10. 任何性能指标越界或造成 APP 崩溃,优酷通用性能测试一招搞定