在讲《21张图讲解集合的线程不安全》那一篇,我留了一个彩蛋,就是Queue(队列)还没有讲,这次我们重点来看看Java中的Queue家族,总共涉及到18种Queue。这篇恐怕是市面上最全最细 讲解Queue的。
本篇主要内容如下:
帮你总结好的阻塞队列:
hi,大家好,我的英文名叫Queue
,中文名叫队列
,无论现实生活中还是计算机的世界中,我都是一个很重要的角色哦~
我是一种数据结构
,大家可以把我想象成一个数组,元素从我的一头进入、从另外一头出去,称为FIFO原则(先进先出原则)。
我还有两个亲兄弟:List
(列表)、Set
(集),他们都是Collection
的儿子,我还有一个远房亲戚:Map
(映射)。他们都是java.util
包这个大家庭的成员哦~
18种队列分为三大类: 接口、抽象类、普通类。
弄清楚下面的继承实现关系对后面理解18种队列有很大帮助。
Queue
接口继承 Collection
接口,Collection
接口继承 Iterable
接口BlockingQueue
接口、Deque
接口 继承 Queue
接口AbstractQueue
抽象类实现 Queue
接口BlockingDeque
接口、TransferQueue
接口继承 BlockingQueue
接口BlockingDeque
接口继承Deque
接口LinkedBlockingDeque
类实现 BlockingDeque
接口LinkedTransferQueue
类接口实现 TransferQueue
接口LinkedList
类、ArrayDeque
类、ConcurrentLinkedDeque
类实现 了Deque
接口ArrayBlockingQueue
类、LinkendBlockingQueue
类、LinkedBlockingDeque
类、LinkedTransferQueue
类、SynchronousQueue
类、PriorityBlockQueue
类、DelayQueue类
继承 了AbstractQueue
抽象类和实现了BlockingQueue接口PriorityQueue
类和ConcurrentLinkedQueue
类继承 了AbstractQueue
抽象类注意:
Queue接口是一种Collection,被设计用于处理之前临时保存在某处的元素。
除了基本的Collection操作之外,队列还提供了额外的插入、提取和检查操作。每一种操作都有两种形式:如果操作失败,则抛出一个异常;如果操作失败,则返回一个特殊值(null或false,取决于是什么操作)。
队列通常是以FIFO(先进先出)的方式排序元素,但是这不是必须的。
只有优先级队列可以根据提供的比较器对元素进行排序或者是采用正常的排序。无论怎么排序,队列的头将通过调用remove()或poll()方法进行移除。在FIFO队列种,所有新的元素被插入到队尾。其他种类的队列可能使用不同的布局来存放元素。
每个Queue必须指定排序属性。
总共有3组方法,每一组方法对应两个方法。如下图所示:
动作 | 抛异常 | 返回特殊值 |
---|---|---|
Insert | add(e) | offer(e) |
Remove | remove() | poll |
Examine | element() | peek() |
(1)比如添加(Insert)
元素的动作,会有两种方式:add(e)
和offer(e)
。如果调用add(e)方法时,添加失败,则会抛异常
,而如果调用的是offer(e)方法失败时,则会返回false
。offer方法用于异常是正常的情况下使用,比如在有界队列中,优先使用offer方法。假如队列满了,不能添加元素,offer方法返回false,这样我们就知道是队列满了,而不是去handle运行时抛出的异常。
(2)同理,移除(Remove)元素的动作,队列为空时,remove方法抛异常,而poll返回null。如果移除头部的元素成功,则返回移除的元素。
(3)同理,检测(Examine)元素的动作,返回头部元素(最开始加入的元素),但不删除元素, 如果队列为空,则element()方法抛异常,而peek()返回false。
(4)Queue接口没有定义阻塞队列的方法,这些方法在BlockQueue接口中定义了。
(5)Queue实现类通常不允许插入null元素,尽管一些实现类比如LinkedList不禁止插入null,但是还是不建议插入null,因为null也被用在poll方法的特殊返回值,以说明队列不包含元素。
(1)Deque概念: 支持两端元素插入和移除的线性集合。名称deque
是双端队列的缩写,通常发音为deck
。大多数实现Deque的类,对它们包含的元素的数量没有固定的限制的,支持有界和***。
(2)Deque方法说明:
**说明: **
该列表包含包含访问deque两端元素的方法,提供了插入,移除和检查元素的方法。
这些方法种的每一种都存在两种形式:如果操作失败,则会抛出异常,另一种方法返回一个特殊值(null或false,取决于具体操作)。
插入操作的后一种形式专门设计用于容量限制的Deque实现,大多数实现中,插入操作不能失败,所以可以用插入操作的后一种形式。
Deque接口扩展了Queue接口,当使用deque作为队列时,作为FIFO。元素将添加到deque的末尾,并从头开始删除。
作为FIFO时等价于Queue的方法如下表所示:
比如Queue的add方法和Deque的addLast方法等价。
Deque也可以用作LIFO(后进先出)栈,这个接口优于传统的Stack类。当作为栈使用时,元素被push到deque队列的头,而pop也是从队列的头pop出来。
Stack(栈)的方法正好等同于Deque的如下方法:
注意:peek方法不论是作为栈还是队列,都是从队列的检测队列的头,返回最先加入的元素。比如第一次put 100,第二次put 200,则peek返回的是100。如下图所示:
AbstractQueue是一个抽象类,继承了Queue接口,提供了一些Queue操作的骨架实现。
方法add、remove、element方法基于offer、poll和peek。也就是说如果不能正常操作,则抛出异常。我们来看下AbstactQueue是怎么做到的。
public boolean add(E e) { if (offer(e)) return true; else throw new IllegalStateException("Queue full"); }
public E remove() { E x = poll(); if (x != null) return x; else throw new NoSuchElementException(); }
public E element() { E x = peek(); if (x != null) return x; else throw new NoSuchElementException(); }
注意:
ArrayBlockingQueue
类、LinkendBlockingQueue
类、LinkedBlockingDeque
类、LinkedTransferQueue
类、SynchronousQueue
类、PriorityBlockQueue
类、DelayQueue类
继承 了AbstractQueue
抽象类PriorityQueue
类和ConcurrentLinkedQueue
类继承 了AbstractQueue
抽象类(1)BlockingQueue(阻塞队列)也是一种队列,支持阻塞的插入和移除方法。
(3)阻塞的插入:当队列满时,队列会阻塞插入元素的线程,直到队列不满。
(4)阻塞的移除:当队列为空,获取元素的线程会等待队列变为非空。
(5)应用场景:生产者和消费者,生产者线程向队列里添加元素,消费者线程从队列里移除元素,阻塞队列时获取和存放元素的容器。
(6)为什么要用阻塞队列:生产者生产和消费者消费的速率不一样,需要用队列来解决速率差问题,当队列满了或空的时候,则需要阻塞生产或消费动作来解决队列满或空的问题。
线程A往阻塞队列(Blocking Queue)中添加
元素,而线程B从阻塞队列中移除
元素。
BlockingQueue接口的10个核心方法:
10个核心方法总结如下:
有三大类操作:插入、移除、检查。
IllegalStateException
- 队列满了ClassCastException
- 添加的元素类型不匹配NullPointerException
- 添加的元素为nullIllegalArgumentException
- 添加的元素某些属性不匹配NoSuchElementException
- 如果这个队列是空的是阻塞队列BlockingQueue
和双向队列Deque
接口的结合。有如下方法:
示例:
尝试执行以下方法:
LinkedBlockingDeque queue = new LinkedBlockingDeque(); queue.addFirst("test1"); queue.addFirst(300); queue.addLast("400");
最后队列中的元素顺序如下:
300, test1, 400。
先添加了test1放到队列的头部,然后在头部的前面放入300,所以300在最前面,成为头部,然后将400放入队列的尾部,所以最后的结果是300, test1, 400。
如果有消费者正在获取元素,则将队列中的元素传递给消费者。如果没有消费者,则等待消费者消费。我把它称作使命必达队列,必须将任务完成才能返回。
transfer(E e)
原理如下图所示:
tryTransfer(E e)
tryTransfer(E e, long timeout, TimeUnit unit)
getWaitingConsumerCount()
hasWaitingConsumer()
PriorityQueue是一个支持优先级的***阻塞队列。
默认自然顺序升序排序。
可以通过构造参数Comparator来对元素进行排序。
public PriorityQueue(Comparator<? super E> comparator) { this(DEFAULT_INITIAL_CAPACITY, comparator); }
public Comparator<? super E> comparator() { return comparator; }
Arrays.sort(pq.toArray)
。offer
、add
)和出列( poll
、remove()
)的时间复杂度O(log(n))。我们来看下节点类Node
private static class Node<E> { E item; //元素 Node<E> next; //向后的节点链接 Node<E> prev; //向前的节点链接 Node(Node<E> prev, E element, Node<E> next) { this.item = element; this.next = next; this.prev = prev; } }
1.LinkedList的增加和删除效率相对较高,而查找和修改的效率相对较低。
2.以下情况建议使用ArrayList
3.以下情况建议使用LinkedList
LinkedList不是线程安全的,所以可以使用如下方式保证线程安全。
List list = Collections.synchronizedList(new LinkedList<>());
LinkedList 继承了 AbstractSequentialList 类。
LinkedList 实现了 Queue 接口,可作为队列使用。
LinkedList 继承了 AbstractQueue抽象类,具有队列的功能。
LinkedList 实现了 List 接口,可进行列表的相关操作。
LinkedList 实现了 Deque 接口,可作为双向队列使用。
LinkedList 实现了 Cloneable 接口,可实现克隆。
LinkedList 实现了 java.io.Serializable 接口,即可支持序列化,能通过序列化去传输。
ConcurrentLinkedQueue queue = new ConcurrentLinkedQueue(); BuildingBlockWithName buildingBlock = new BuildingBlockWithName("三角形", "A"); concurrentLinkedQueue.add(buildingBlock);
创建一个ArrayDeque,往arrayDeque队尾添加元素。
ArrayDeque arrayDeque = new ArrayDeque(); for (int i = 0; i < 50; i++) { arrayDeque.add(buildingBlock); // add方法等价于addLast方法 }
创建两个积木:三角形、四边形,然后添加到队列:
BuildingBlockWithName buildingBlock1 = new BuildingBlockWithName("三角形", "A"); BuildingBlockWithName buildingBlock2 = new BuildingBlockWithName("四边形", "B"); ConcurrentLinkedDeque concurrentLinkedDeque = new ConcurrentLinkedDeque(); concurrentLinkedDeque.addFirst(buildingBlock1); concurrentLinkedDeque.addLast(buildingBlock2); //结果:顺序:三角形、四边形
创建两个积木:三角形、四边形,然后添加到队列:
BuildingBlockWithName buildingBlock1 = new BuildingBlockWithName("三角形", "A"); BuildingBlockWithName buildingBlock2 = new BuildingBlockWithName("四边形", "B"); ArrayBlockingQueue arrayBlockingQueue = new ArrayBlockingQueue(100, true); arrayBlockingQueue.add(buildingBlock1); arrayBlockingQueue.add(buildingBlock2);
LinkedList linkedList1 = new LinkedList(); linkedList1.add("A"); linkedList1.add("B"); linkedList1.add("C");
LinkedBlockingDeque可以用在“工作窃取“模式中。
工作窃取算法
:某个线程比较空闲,从其他线程的工作队列中的队尾窃取任务来帮忙执行。
LinkedTransferQueue = 阻塞队列+链表结构+TransferQueue
之前我们讲“使命必达TransferQueue接口时已经介绍过了TransferQueue接口 ,所以LinkedTransferQueue接口跟它相似,只是加入了阻塞插入和移除的功能,以及结构是链表结构。
之前的TransferQueue也讲到了3个案例来说明TransferQueue的原理,大家可以回看TransferQueue。
生产者1 transfer A 生产者2 transfer D 2s后... 消费者 take A 生产者1 put B 2s后... 消费者 take D 生产者2 transfer E 2s后... 消费者 take B 生产者1 transfer C
(1)首先生产者线程1和2 调用transfer方法来传输A和D,发现没有消费者线程接收,所以被阻塞。
(2)消费者线程过了2s后将A拿走了,然后生产者1 被释放继续执行,传输元素B,发现又没有消费者消费,所以进行了等待。
(3)消费者线程过了2s后,将排在队列首部的D元素拿走,生产者2继续往下执行,传输元素E,发现没有消费者,所以进行了等待。
(4)消费者线程过了2s后,将排在队列首部的B元素拿走,生产者1传输C元素,等待消费者拿走。
(5)消费者不再消费了,所以生产者1和生产者2都被阻塞了,元素C和,元素E都没有被拿走,而且生产者2的F元素还没有开始传输,因为在等待元素D被拿走。
(6)看下队列里面确实有C和E元素,而且E排在队列的首部。
我称SynchronousQueue为”传球好手“。想象一下这个场景:小明抱着一个篮球想传给小花,如果小花没有将球拿走,则小明是不能再拿其他球的。
SynchronousQueue负责把生产者产生的数据传递给消费者线程。
SynchronousQueue本身不存储数据,调用了put方法后,队列里面也是空的。
每一个put操作必须等待一个take操作完成,否则不能添加元素。
适合传递性场景。
性能高于ArrayBlockingQueue和LinkedBlockingQueue。
我们创建了两个线程,一个线程用于生产,一个线程用于消费
t1 put A t2 take A 5秒后... t1 put B t2 take B 5秒后... t1 put C t2 take C
小结:说明生产线程执行put第一个元素"A" 操作后,需要等待消费者线程take完“A”后,才能继续往下执行代码。
public class DelayQueue<E extends Delayed> extends AbstractQueue<E> implements BlockingQueue<E> {
public boolean offer(E e, long timeout, TimeUnit unit) { return offer(e); }
if (first == null || first.getDelay(NANOSECONDS) > 0) return null; else return q.poll();
缓存系统的设计:可以用DelayQueue保存缓存元素的有效期。然后用一个线程循环的查询DelayQueue队列,一旦能从DelayQueue中获取元素时,表示缓存有效期到了。
定时任务调度:使用DelayQueue队列保存当天将会执行的任务和执行时间,一旦从DelayQueue中获取到任务就开始执行。比如Java中的TimerQueue就是使用DelayQueue实现的。
这一篇花了很多心思在上面,看官方英文文档、画原理图、写demo代码,排版。这恐怕是市面上最全最细讲解Queue的。
❝
你好,我是
悟空哥
,「7年项目开发经验,全栈工程师,开发组长,超喜欢图解编程底层原理」。正在编写两本PDF,分别是 1、Spring Cloud实战项目(佳必过),2、Java并发必知必会。我还手写了2个小程序
,Java刷题小程序,PMP刷题小程序,点击我的公众号菜单打开!另外有111本架构师资料以及1000道Java面试题,都整理成了PDF,可以关注公众号 「悟空聊架构」 回复悟空
领取优质资料。❞
「转发->在看->点赞->收藏->评论!!!」 是对我最大的支持!