文章目录
ArrayDeque 简介
首先在Java的集合框架中,要知道:
- Stack是一种先进后出的数据结构:栈;
- Queue是一种先进先出(First In First Out)的数据结构:队列。
Deque是Queue接口的子接口,它代表一个双端队列,内部定义了一系列双端队列的方法。而ArrayQueue是Deque接口的一个典型实现类,它是一个基于数组实现的双端队列,并且其还是通过循环数组的方式实现的循环队列,并通过位运算来提高效率,容量大小始终是2的次幂。
所以当需要使用栈时,Java已不推荐使用Stack,而是推荐使用更高效的ArrayDeque;而且Queue只是一个接口,当需要使用队列时也就首选ArrayDeque了(次选是LinkedList)。
- Deque与Queue相对应的接口
- Deque与Stack对应的接口
先总结下:
- ArrayDeque是采用数组方式实现的双端队列;
- ArrayDeque的出队入队是通过头尾指针循环利用数组实现的;
- ArrayDeque容量不足时是会扩容的,每次扩容容量增加一倍;
- ArrayDeque可以直接作为栈使用。
一、实现原理
ArrayDeque底层是通过数组+两个索引指针head、tail来实现的,而为了满足可以同时在数组两端插入或删除元素的需要,该数组还必须是循环的,即循环数组。所以规定head指向首端第一个有效元素,tail指向尾端第一个可以插入元素的空位的,而且head不一定总等于0,tail也不一定总比head大。这样ArrayDeque既可以直接向尾部添加数据又可以直接向头部之前添加数据,并且不需要大面积地移动数据。
二、源码分析
2.1 继承与实现关系
- 继承了AbstractCollection,表明它提供了对集合操作的基本实现;
- 实现了Deque接口,意味着它是一个双端队列,支持首尾进出操作;
- 实现了Cloneable接口,意味着它能被克隆;
- 实现了java.io.Serializable接口,意味着它支持序列化。
2.2 重要成员信息
- elements:用于存储ArrayDeque的元素的数组缓冲区,是ArrayDequet的底层数据结构,用于存放实际元素;
- head:指向首端第一个有效元素的索引指针;
- tail:指向尾端第一个可以插入元素的空位的索引指针;
- MAX_ARRAY_SIZE:数组缓冲区的最大容量。
2.3 构造方法
- ArrayDeque(),创建了一个长度为16的数组 elements。即ArrayDeque的容量为 16。
- ArrayDeque(int numElements),创建了一个指定大小(最大不超过最大容量,最小不低于1)的elements;
2.4 扩容机制
- 容器容量在不大于64的情况下进行扩容,扩容2倍单位大小,否则扩容2个单位。所以一般情况下,数组的长度总是 2 的整数次方;
- 当head 大于tail或head等于tail且不为空的时候,我们则要采用段复制,最终保证<head --> tail>可以形成一个不间断连续段。
2.5 重要方法
2.5.1 双端队列方法
源码:
2.5.2 单端队列方法
一端进另一端出
源码:
2.5.3 堆栈方法
一端操作先进先出
源码:
2.5.4 重要方法
- addFirst(E e):它的作用是在Deque的首端插入元素,也就是在head的前面插入元素,在空间足够且下标没有越界的情况下,只需要将elements[–head] = e即可。
如果head为0之后接着调用addFirst(),虽然空余空间还够用,但head为-1,下标越界了。此时处理办法:head 移动到数组尾部
- addLast(E e):它的作用是在Deque的尾端插入元素,也就是在tail的位置插入元素,由于tail总是指向下一个可以插入的空位,因此只需要elements[tail] = e;即可。
插入完成后,tail移动,并检查空间,如果空间已经用光,进行扩容。