Collection - Stack & Queue 源码解析

Stack & Queue概述

Java里有一个叫做Stack的类,却没有叫做Queue的类(它是个接口名字)。当需要使用栈时,Java已不推荐使用Stack,而是推荐使用更高效的ArrayDeque;既然Queue只是一个接口,当需要使用队列时也就首选ArrayDeque了(次选是LinkedList)。

Queue

Queue接口继承自Collection接口,除了最基本的Collection的方法之外,它还支持额外的insertion, extraction和inspection操作。这里有两组格式,共6个方法,一组是抛出异常的实现;另外一组是返回值的实现(没有则返回null)。

Throws exception

Returns special value

Insert

add(e)

offer(e)

Remove

remove()

poll()

Examine

element()

peek()

Deque

Deque​是"double ended queue", 表示双向的队列,英文读作"deck". Deque 继承自 Queue接口,除了支持Queue的方法之外,还支持insert​, remove​和examine操作,由于Deque是双向的,所以可以对队列的头和尾都进行操作,它同时也支持两组格式,一组是抛出异常的实现;另外一组是返回值的实现(没有则返回null)。共12个方法如下:

First Element - Head

Last Element - Tail

Throws exception

Special value

Throws exception

Special value

Insert

addFirst(e)

offerFirst(e)

addLast(e)

offerLast(e)

Remove

removeFirst()

pollFirst()

removeLast()

pollLast()

Examine

getFirst()

peekFirst()

getLast()

peekLast()

当把Deque​当做FIFO的queue​来使用时,元素是从deque​的尾部添加,从头部进行删除的; 所以deque​的部分方法是和queue是等同的。具体如下:

Queue Method

Equivalent Deque Method

add(e)

addLast(e)

offer(e)

offerLast(e)

remove()

removeFirst()

poll()

pollFirst()

element()

getFirst()

peek()

peekFirst()

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个之多,但无非就是对容器的两端进行操作,或添加,或删除,或查看。明白了这一点讲解起来就会非常简单。

ArrayDeque和LinkedList是Deque的两个通用实现,由于官方更推荐使用AarryDeque用作栈和队列,加之上一篇已经讲解过LinkedList,本文将着重讲解ArrayDeque的具体实现。

从名字可以看出ArrayDeque底层通过数组实现,为了满足可以同时在数组两端插入或删除元素的需求,该数组还必须是循环的,即循环数组(circular array),也就是说数组的任何一点都可能被看作起点或者终点。ArrayDeque是非线程安全的(not thread-safe),当多个线程同时使用的时候,需要程序员手动同步;另外,该容器不允许放入null元素。


方法剖析

addFirst()

addFirst(E e)​的作用是在Deque的首端插入元素,也就是在head​的前面插入元素,在空间足够且下标没有越界的情况下,只需要将elements[--head] = e即可。

//addFirst(E e)
public void addFirst(E e) {
if (e == null)//不允许放入null
throw new NullPointerException();
elements[head = (head - 1) & (elements.length - 1)] = e;//2.下标是否越界
if (head == tail)//1.空间是否够用
doubleCapacity();//扩容
}

上述代码我们看到,空间问题是在插入之后解决的,因为tail​总是指向下一个可插入的空位,也就意味着elements数组至少有一个空位,所以插入元素的时候不用考虑空间问题。

下标越界的处理解决起来非常简单,head = (head - 1) & (elements.length - 1)​就可以了,这段代码相当于取余,同时解决了​head​为负值的情况。因为elements.length​必需是2​的指数倍,elements - 1​就是二进制低位全1​,跟head - 1​相与之后就起到了取模的作用,如果head - 1​为负数(其实只可能是-1),则相当于对其取相对于elements.length的补码。

addLast()

addLast(E e)​的作用是在Deque的尾端插入元素,也就是在tail​的位置插入元素,由于tail​总是指向下一个可以插入的空位,因此只需要elements[tail] = e;​即可。插入完成后再检查空间,如果空间已经用光,则调用doubleCapacity()进行扩容。

public void addLast(E e) {
if (e == null)//不允许放入null
throw new NullPointerException();
elements[tail] = e;//赋值
if ( (tail = (tail + 1) & (elements.length - 1)) == head)//下标越界处理
doubleCapacity();//扩容
}

pollFirst()

pollFirst()​的作用是删除并返回Deque首端元素,也即是head​位置处的元素。如果容器不空,只需要直接返回elements[head]​即可,当然还需要处理下标的问题。由于ArrayDeque​中不允许放入null​,当elements[head] == null时,意味着容器为空。

public E pollFirst() {
E result = elements[head];
if (result == null)//null值意味着deque为空
return null;
elements[h] = null;//let GC work
head = (head + 1) & (elements.length - 1);//下标越界处理
return result;
}

pollLast()

pollLast()​的作用是删除并返回Deque尾端元素,也即是tail位置前面的那个元素。

public E pollLast() {
int t = (tail - 1) & (elements.length - 1);//tail的上一个位置是最后一个元素
E result = elements[t];
if (result == null)//null值意味着deque为空
return null;
elements[t] = null;//let GC work
tail = t;
return result;
}