问题
(1)什么是双端队列?
(2)ArrayDeque是怎么实现双端队列的?
(3)ArrayDeque是线程安全的吗?
(4)ArrayDeque是有界的吗?
简介
双端队列是一种特殊的队列,它的两端都可以进出元素,故而得名双端队列。
ArrayDeque是一种以数组方式实现的双端队列,它是非线程安全的。
UML继承关系
通过继承体系可以看,ArrayDeque实现了Deque接口,Deque接口继承自Queue接口,它是对Queue的一种增强。
1 |
|
Deque中新增了以下几类方法:
(1)*First,表示从队列头操作元素;
(2)*Last,表示从队列尾操作元素;
(3)push(e),pop(),以栈的方式操作元素的方法;
源码分析
主要属性
1 | // 存储元素的数组 |
从属性我们可以看到,ArrayDeque使用数组存储元素,并使用头尾指针标识队列的头和尾,其最小容量是8。
主要构造方法
1 | // 默认构造方法,初始容量为16 |
通过构造方法,我们知道默认初始容量是16,最小容量是8。
入队
入队有很多方法,我们这里主要分析两个,addFirst(e)和addLast(e)。
1 | // 从队列头入队 |
(1)入队有两种方式,从队列头或者从队列尾;
(2)如果容量不够了,直接扩大为两倍;
(3)通过取模的方式让头尾指针在数组范围内循环;
(4)x & (len - 1) = x % len,使用&的方式更快;
扩容
1 | private void doubleCapacity() { |
扩容这里迁移元素可能有点绕,请看下面这张图来理解。
出队
出队同样有很多方法,我们主要看两个,pollFirst()和pollLast()。
1 | // 从队列头出队 |
(1)出队有两种方式,从队列头或者从队列尾;
(2)通过取模的方式让头尾指针在数组范围内循环;
(3)出队之后没有缩容哈哈^^
栈
前面我们介绍Deque的时候说过,Deque可以直接作为栈来使用,那么ArrayDeque是怎么实现的呢?
1 | public void push(E e) { |
是不是很简单,入栈出栈只要都操作队列头就可以了。
总结
(1)ArrayDeque是采用数组方式实现的双端队列;
(2)ArrayDeque的出队入队是通过头尾指针循环利用数组实现的;
(3)ArrayDeque容量不足时是会扩容的,每次扩容容量增加一倍;
(4)ArrayDeque可以直接作为栈使用;