Java容器解析系列(6) Queue Deque AbstractQueue 详解

首先我们来看一下Queue接口:
/**
* @since 1.5
*/
public interface Queue<E> extends Collection<E> {
// 添加指定元素,在添加失败时(队列满),抛出IllegalStateException
boolean add(E e);
// 添加指定元素,在添加失败时(队列满),返回false
boolean offer(E e);
// 删除并返回头部,当队列为空时,抛出NoSuchElementException
E remove();
// 删除并返回头部,当队列为空时,返回null
E poll();
// 获取但不删除头部,当队列为空时,抛出NoSuchElementException
E element();
// 获取但不删除头部,队列为空时,返回null
E peek();
}
javadoc中对于Queue接口的表述如下:
Queue 用来存放 优先处理元素 的集合,这种场景一般用于缓冲、并发访问。
除了继承 Collection 接口的一些方法,Queue 还添加了额外的 添加、删除、查询操作。
添加、删除、查询这些个操作都提供了两种形式,其中一种在操作失败时直接抛出异常,而另一种则返回一个特殊值----false或null;后一种是特地提供给容量受限的实现类的;
(具体哪些方法抛出异常,哪些方法返回特殊值见下图)
Queue 一般都是 FIFO 的,但是也有例外,比如优先队列 priority queue(元素顺序是根据自然排序或者自定义 comparator 排序);
再比如 LIFO 的队列(跟栈一样,后进先出)。
不论进入、出去的先后顺序是怎样的,使用 remove(),poll() 方法操作的都是 头部 的元素;
而插入的位置则不一定是在队尾了,不同的 queue 会有不同的插入逻辑。
一般情况下 Queue 的实现类不允许 null 元素,因为null被poll()/peek()方法作为一个特殊返回值,以此来表示队列为空,尽管一些实现类允许null,比如LinkedList;
从上面的描述我们可以看出,Java中的Queue与我们在数据结构中理解的队列不一样:数据结构中的队列只允许FIFO,而Java中的Queue接口表示其可以是一个队列,但是其不必须是一个队列;
接下来我们来看Deque接口:
/**
* @since 1.6
*/
public interface Deque<E> extends Queue<E> {
// 添加指定元素到队头,如果失败,将抛出IllegalStateException;
// 如果是容量受限的队列,建议使用offerFirst();
void addFirst(E e);
// 添加指定元素到队尾,如果失败,将抛出IllegalStateException;
// 如果是容量受限的队列,建议使用offerLast();
// 该方法和add()相同;
void addLast(E e);
// 添加指定元素到队头,如果失败,返回false;
boolean offerFirst(E e);
// 添加指定元素到队尾,如果失败,返回false;
// 该方法与offer()相同;
boolean offerLast(E e);
// 删除并返回头部,当队列为空时,抛出NoSuchElementException;
// 该方法与remove()相同;
E removeFirst();
// 删除并返回尾部,当队列为空时,抛出NoSuchElementException;
E removeLast();
// 删除并返回头部,当队列为空时,返回null;
// 该方法与poll()相同;
E pollFirst();
// 删除并返回尾部,当队列为空时,返回null;
E pollLast();
// 获取但不删除头部,当队列为空时,抛出NoSuchElementException;
// 该方法与element()相同;
E getFirst();
// 获取但不删除队尾,当队列为空时,抛出NoSuchElementException;
E getLast();
// 获取但不删除头部,队列为空时,返回null;
// 该方法与peek()相同;
E peekFirst();
// 获取但不删除队尾,队列为空时,返回null;
E peekLast();
boolean removeFirstOccurrence(Object o);
boolean removeLastOccurrence(Object o);
// *** Queue methods ***
// 队列接口方法
boolean add(E e);
boolean offer(E e);
E remove();
E poll();
E element();
E peek();
// *** Stack methods ***
// 栈相关方法
// 将指定元素入栈;
// 该方法与addFirst()相同;
void push(E e);
// 弹栈;
// 该方法与removeFirst()相同;
E pop();
// *** Collection methods ***
// Collection接口方法
boolean remove(Object o);
boolean contains(Object o);
public int size();
Iterator<E> iterator();
Iterator<E> descendingIterator();
}
javadoc中对于Deque接口的表述如下:
Deque /dek/ :双端队列 Double ended queue
大多数双端队列大小不确定,但是 Deque 支持容量受限的双端队列。
Deque 接口定义了一些从头部和尾部访问元素的方法。比如分别在头部、尾部进行插入、删除、获取元素。
每个操作都有两种方法,一种在异常情况下直接抛出异常奔溃,另一种则不会抛异常,而是返回特殊的值--false或null;后一种是特地提供给容量受限的实现类的;
Deque接口继承自Queue,除了添加了双端队列的方法外,还定义了栈操作的方法,java.util.Stack推荐使用Deque进行栈操作;
关于Queue和Deque的一些联系和区别,可以总结如下:
和Queue一样,增删改查,每个操作都有两种方法,一种在异常情况下直接抛出异常奔溃,另一种则不会抛异常,而是返回特殊的值---false或null:

Deque中有些新添加的方法与Queue中的某些方法预期是相等的(同一种实现,不同的方法名):

栈相关方法也是通过Deque新添加的双端队列相关方法实现(同一种实现,不同的方法名):

接下来我们来看AbstractQueue:
/**
* @since 1.5
*/
public abstract class AbstractQueue<E> extends AbstractCollection<E> implements Queue<E> {
protected AbstractQueue() {
}
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();
}
public void clear() {
while (poll() != null)
;
}
public boolean addAll(Collection<? extends E> c) {
// 不允许添加null
if (c == null)
throw new NullPointerException();
// 不允许添加自身
if (c == this)
throw new IllegalArgumentException();
boolean modified = false;
for (E e : c)
if (add(e))
modified = true;
return modified;
}
}
从源码可以看出:
- 该类提供Queue的骨架实现;
- 方法 add() ,remove(),和element()分别基于offer(),poll()和peek()实现;只是在失败时抛出异常而已,而其子类就要具体实现这些方法了;
- 继承该类的实现类不应该允许插入null;
Java容器解析系列(6) Queue Deque AbstractQueue 详解的更多相关文章
- Java容器解析系列(4) ArrayList Vector Stack 详解
ArrayList 这里关于ArrayList本来都读了一遍源码,并且写了一些了,突然在原来的笔记里面发现了收藏的有相关博客,大致看了一下,这些就是我要写的(╹▽╹),而且估计我还写不到博主的水平,这 ...
- Java容器解析系列(0) 开篇
最近刚好学习完成数据结构与算法相关内容: Data-Structures-and-Algorithm-Analysis 想结合Java中的容器类加深一下理解,因为之前对Java的容器类理解不是很深刻, ...
- Java容器解析系列(11) HashMap 详解
本篇我们来介绍一个最常用的Map结构--HashMap 关于HashMap,关于其基本原理,网上对其进行讲解的博客非常多,且很多都写的比较好,所以.... 这里直接贴上地址: 关于hash算法: Ha ...
- Java容器解析系列(7) ArrayDeque 详解
ArrayDeque,从名字上就可以看出来,其是通过数组实现的双端队列,我们先来看其源码: /** 有自动扩容机制; 不是线程安全的; 不允许添加null; 作为栈使用时比java.util.Stac ...
- Java容器解析系列(9) PrioriyQueue详解
PriorityQueue:优先级队列; 在介绍该类之前,我们需要先了解一种数据结构--堆,在有些书上也直接称之为优先队列: 堆(Heap)是是具有下列性质的完全二叉树:每个结点的值都 >= 其 ...
- Java容器解析系列(1) 迭代的进化——从Enumeration到Iterator
在Java中,对于所有的Collection,都有一个特性,可以通过迭代器来遍历和删除其中的元素,因为Collection接口继承自Iterable接口. public interface Colle ...
- Java容器解析系列(13) WeakHashMap详解
关于WeakHashMap其实没有太多可说的,其与HashMap大致相同,区别就在于: 对每个key的引用方式为弱引用; 关于java4种引用方式,参考java Reference 网上很多说 弱引用 ...
- Java容器解析系列(10) Map AbstractMap 详解
前面介绍了List和Queue相关源码,这篇开始,我们先来学习一种java集合中的除Collection外的另一个分支------Map,这一分支的类图结构如下: 这里为什么不先介绍Set相关:因为很 ...
- Java容器解析系列(2) 具体化的第一步——Collection到AbstractCollection
在通向具体化的List,Queue之前,我们需要先了解一下Collection接口和AbstractCollection抽象类,这两个都是处于Collection顶层的存在. Collection接口 ...
随机推荐
- C#中的装箱(inboxing)和拆箱(unboxing)(简单理解)
装箱和拆箱是值类型和引用类型之间相互转换是要执行的操作. 装箱:将一个值类型隐式地转换成一个object类型,或把这个值类型转换成一个被该值类型应用的接口类型,把一个值类型的值装箱,就是创建一个ob ...
- 进程间通信之信号量、消息队列、共享内存(system v的shm和mmap)+信号signal
进程间通信方式有:System v unix提供3种进程间通信IPC:信号量.消息队列.共享内存.此外,传统方法:信号.管道.socket套接字. [注意上述6种方式只能用户层进程间通信.内核内部有类 ...
- GO语言常量和变量
标识符与关键字 标识符 人为定义有特殊意义的词,Go语言中标识符由字母数字和_(下划线)组成,并且只能以字母和_开头. 关键字 关键字是指编程语言中预先定义好的具有特殊含义的标识符. GO语言中有25 ...
- 各版本最新的Visual C++可再发行组件包(Redistributable Package)下载和合集
Microsoft Visual C++ 2005 Redistributable Package (x86):Microsoft Visual C++ 2005 可再发行组件包 (x86):http ...
- 20190403vim编辑器week1_day3
vi编辑器 作用:编辑文本文件中的内容的工具 命令历史 末行模式中,以:和/开头的命令都有历史纪录,可以首先键入:或/然后按上下箭头来选择某个历史命令. 启动vim 在命令行窗口中输入以下命令即可 v ...
- 1-趣味解读DNS工作原理——转载疯猫网络科技
因为只要我们输入百度.腾讯.淘宝的名字,无论它们的服务器在哪里,历经多少轮查询,我们都能找到并访问之.这就是计算机网络中著名的域名系统DNS(Domain Name System),它能实现把一个网站 ...
- 批处理学习之Bat命令——获取当前盘符、当前目录、上级目录
命令 当前盘符:%~d0 当前路径:%cd% 当前执行命令行:%0 当前bat文件路径:%~dp0 当前bat文件短路径:%~sdp0 测试 下载testBatPath.bat测试文件,双击.bat运 ...
- ssm回顾笔记(一)
这两天来到了农银,这边即将进行的一个项目是将ssh框架的电商项目迁移到springboot+ssm框架上,所以我基本上是三门技术在同时进行学习,当然以前学过ssm,现在只是回顾. spring 注解 ...
- CSS3 傻傻分不清楚的transition, transform 和 animation
transition transition允许css的属性值在一定的时间区间内平滑地过渡,语法如下: transition : transition-property transition-durat ...
- 微信端修改title
function setTitle(t) { document.title = t; var i = document.createElement('iframe'); i.src = "i ...
