ArrayDeque源码剖析
前言
Java里有一个叫做Stack的类,却没有叫做Queue的类(它是个接口名字)。当需要使用栈时,Java已不推荐使用Stack,而是推荐使用更高效的ArrayDeque;既然Queue只是一个接口,当需要使用队列时也就首选ArrayDeque了(次选是LinkedList)。
总体介绍
要讲栈和队列,首先要讲Deque接口。Deque的含义是“double ended queue”,即双端队列,它既可以当作栈使用,也可以当作队列使用。
下表列出了Deque与Queue相对应的接口:
| Queue Method | Equivalent Deque Method | 说明 | 
|---|---|---|
| add(e) | addLast(e) | 向队尾插入元素,失败则抛出异常 | 
| offer(e) | offerLast(e) | 向队尾插入元素,失败则返回 false | 
| remove() | removeFirst() | 获取并删除队首元素,失败则抛出异常 | 
| poll() | pollFirst() | 获取并删除队首元素,失败则返回 null | 
| element() | getFirst() | 获取但不删除队首元素,失败则抛出异常 | 
| peek() | peekFirst() | 获取但不删除队首元素,失败则返回 null | 
下表列出了Deque与Stack对应的接口:
| Stack Method | Equivalent Deque Method | 说明 | 
|---|---|---|
| push(e) | addFirst(e) | 向栈顶插入元素,失败则抛出异常 | 
| 无 | offerFirst(e) | 向栈顶插入元素,失败则返回 false | 
| pop() | removeFirst() | 获取并删除栈顶元素,失败则抛出异常 | 
| 无 | pollFirst() | 获取并删除栈顶元素,失败则返回 null | 
| peek() | getFirst() | 获取但不删除栈顶元素,失败则抛出异常 | 
| 无 | peekFirst() | 获取但不删除栈顶元素,失败则返回 null | 
上面两个表共定义了Deque的12个接口。添加,删除,取值都有两套接口,它们功能相同,区别是对失败情况的处理不同。一套接口遇到失败就会抛出异常,另一套遇到失败会返回特殊值( false 或 null )。除非某种实现对容量有限制,大多数情况下,添加操作是不会失败的。虽然Deque的接口有12个之多,但无非就是对容器的两端进行操作,或添加,或删除,或查看。
| insert | remove | examine | 
|---|---|---|
| addFirst(e) | removeFirst() | getFirst() | 
| offerFirst(e) | pollFirst() | peekFirst() | 
| addLast(e) | removeLast() | getLast() | 
| offerLast(e) | pollLast() | peekLast() | 
ArrayDeque和LinkedList是Deque的两个通用实现,官方更推荐使用AarryDeque用作栈和队列。
从名字可以看出ArrayDeque底层通过数组实现,为了满足可以同时在数组两端插入或删除元素的需求,该数组还必须是循环的,即循环数组(circular array),也就是说数组的任何一点都可能被看作起点或者终点。ArrayDeque是非线程安全的(not thread-safe),当多个线程同时使用的时候,需要程序员手动同步;另外,该容器不允许放入 null 元素。
 
上图中我们看到,head 指向首端第一个有效元素,tail 指向尾端第一个可以插入元素的空位。因为是循环数组,所以 head 不一定总等于0,tail 也不一定总是比 head 大。
| 1 | transient Object[] elements; // non-private to simplify nested class access | 
方法剖析
addFirst()
addFirst(E e) 的作用是在Deque的首端插入元素,也就是在 head 的前面插入元素,在空间足够且下标没有越界的情况下,只需要将 elements[--head] = e 即可。
 
实际需要考虑:1.空间是否够用,2.下标是否越界的问题。上图中,如果 head 为 0 之后接着调用 addFirst() ,虽然空余空间还够用,但 head 为 -1 ,下标越界了。下列代码很好的解决了这两个问题。
| 1 | //addFirst(E e) | 
上述代码我们看到,空间问题是在插入之后解决的,因为 tail 总是指向下一个可插入的空位,也就意味着 elements 数组至少有一个空位,所以插入元素的时候不用考虑空间问题。
下标越界的处理解决起来非常简单,head = (head - 1) & (elements.length - 1) 就可以了,这段代码相当于取余,同时解决了 head 为负值的情况。因为 elements.length 必需是 2 的指数倍,elements - 1 就是二进制低位全 1 ,跟 head - 1 相与之后就起到了取模的作用,如果 head - 1 为负数(其实只可能是 -1 ),则相当于对其取相对于 elements.length 的补码。
下面再说说扩容函数 doubleCapacity() ,其逻辑是申请一个更大的数组(原数组的两倍),然后将原数组复制过去。过程如下图所示:
 
图中我们看到,复制分两次进行,第一次复制 head 右边的元素,第二次复制 head 左边的元素。
| 1 | //doubleCapacity() | 
addLast()
addLast(E e) 的作用是在Deque的尾端插入元素,也就是在 tail 的位置插入元素,由于 tail 总是指向下一个可以插入的空位,因此只需要 elements[tail] = e; 即可。插入完成后再检查空间,如果空间已经用光,则调用 doubleCapacity() 进行扩容。
 
| 1 | public void addLast(E e) { | 
pollFirst()
pollFirst() 的作用是删除并返回Deque首端元素,也即是 head 位置处的元素。如果容器不空,只需要直接返回 elements[head] 即可,当然还需要处理下标的问题。由于ArrayDeque中不允许放入 null,当 elements[head] == null 时,意味着容器为空。
| 1 | public E pollFirst() { | 
pollLast()
pollLast() 的作用是删除并返回Deque尾端元素,也即是 tail 位置前面的那个元素。
| 1 | public E pollLast() { | 
peekFirst()
peekFirst() 的作用是返回但不删除Deque首端元素,也即是head位置处的元素,直接返回 elements[head]即可。
| 1 | public E peekFirst() { | 
peekLast()
peekLast() 的作用是返回但不删除Deque尾端元素,也即是 tail位置前面的那个元素。
| 1 | public E peekLast() { |