Queue与Deque的区别

Queue与Deque的区别​ 在研究java集合源码的时候,发现了一个很少用但是很有趣的点:Queue以及Deque,平常在写leetcode经常用LinkedList向

大家好,欢迎来到IT知识分享网。

​ 在研究java集合源码的时候,发现了一个很少用但是很有趣的点:Queue以及Deque,平常在写leetcode经常用LinkedList向上转型Deque作为栈或者队列使用,但是一直都不知道Queue的作用,于是就直接官方文档好了。

Queue与Deque的区别

从上图看出,Queue以及Deque都是继承于Collection,Deque是Queue的子接口。

下面来看一下官方文档的解释。

A linear collection that supports element insertion and removal at both ends. The name deque is short for “double ended queue” and is usually pronounced “deck”. Most Deque implementations place no fixed limits on the number of elements they may contain, but this interface supports capacity-restricted deques as well as those with no fixed size limit.

A collection designed for holding elements prior to processing. Besides basic Collection operations, queues provide additional insertion, extraction, and inspection operations. Each of these methods exists in two forms: one throws an exception if the operation fails, the other returns a special value (either null or false, depending on the operation). The latter form of the insert operation is designed specifically for use with capacity-restricted Queue implementations; in most implementations, insert operations cannot fail.

从Deque的解释中,我们可以得知:Deque是double ended queue,我将其理解成双端结束的队列,双端队列,可以在首尾插入或删除元素。而Queue的解释中,Queue就是简单的FIFO队列。

所以在概念上来说,Queue是FIFO的单端队列,Deque是双端队列。

而在使用上,又有什么差别呢?

§ 使用

从上图我们可以得知,Queue有一个直接子类PriorityQueue,而Deque中直接子类有两个:LinkedList以及ArrayDeque。

Queue与Deque的区别

我觉得重点就在圈定的两个单词:无边界的,优先级的堆。然后再看看源码

Queue与Deque的区别

Queue与Deque的区别

在第一张图片的源码中,明显看到PriorityQueue的底层数据结构是数组,而无边界的形容,那么指明了PriorityQueue是自带扩容机制的,具体请看PriorityQueue的grow方法。

在第二张第三张图片中,可以看到插入元素的时候是需要经过compareTo的处理,那么最常用就是一些范围极值的输出,类似于堆排序的用法。

下面演示一下正反序输出三个元素的使用

private static void negativePrint(int[] nums) { PriorityQueue<Integer> queue=new PriorityQueue<>(new Comparator<Integer>() { @Override public int compare(Integer o1, Integer o2) { return o2-o1; } }); for(int temp:nums){ queue.add(temp); } System.out.println(); System.out.print("倒序输出:"); for(int i=0;i<3;i++){ System.out.print(queue.poll()+" "); } } private static void positivePrint(int[] nums){ PriorityQueue<Integer> queue=new PriorityQueue<>(); for(int temp:nums){ queue.add(temp); } System.out.print("正序输出:"); for(int i=0;i<3;i++){ System.out.print(queue.poll()+" "); } } 
正序输出:1 2 3 倒序输出:9 8 8 

这个在一些排行榜或者输入第N个最大/小元素会比较常用。

  • LinkedList以及ArrayDeque
Queue与Deque的区别

从官方解释来看,ArrayDeque是无初始容量的双端队列,LinkedList则是双向链表。而我们还能看到,ArrayDeque作为队列时的效率比LinkedList要高,而在栈的使用场景下,无疑具有尾结点不需判空的LinkedList较高效。

下面演示ArrayDeque作为队列以及LinkedList作为栈的使用

private static void usingAsQueue() { Deque<Integer> queue=new ArrayDeque<>(); System.out.println("队列为空:"+queue.isEmpty()); //判断队列是否为空 queue.addLast(12); //添加元素 System.out.println("队列为空:"+queue.isEmpty()); //判断队列是否为空 System.out.println(queue.peekFirst()); //获取队列首部元素 System.out.println(queue.pollFirst()); //获取并移除栈顶元素 System.out.println("队列为空:"+queue.isEmpty()); //判断队列是否为空 } private static void usingAsStack() { //作为栈使用 Deque<Integer> stack=new LinkedList<>(); System.out.println("栈为空:"+stack.isEmpty()); //判断栈是否为空 stack.addFirst(12); System.out.println("栈为空:"+stack.isEmpty()); //判断栈是否为空 System.out.println(stack.peekFirst()); //获取栈顶元素 System.out.println(stack.pollFirst()); //获取并移除栈顶元素 System.out.println("栈为空:"+stack.isEmpty()); //判断栈是否为空 System.out.println("============================================"); } 

其实ArrayDeque和LinkedList都可以作为栈以及队列使用,但是从执行效率来说,ArrayDeque作为队列以及LinkedList作为栈使用会是更好的选择。

另外,我在leetcode看到有人采用Vector下的Stack,这个同步加锁粒度过大(对象级),另外我觉得算法中没有线程同步的需要吧。

小结

PriorityQueue可以作为堆使用,而且可以根据传入的Comparator实现大小的调整,会是一个很好的选择。

ArrayDeque通常作为栈或队列使用,但是栈的效率不如LinkedList高。

LinkedList通常作为栈或队列使用,但是队列的效率不如ArrayQueue高。

免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://yundeesoft.com/79865.html

(0)

相关推荐

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注

关注微信