ArrayDeque介绍


ArrayDeque是一个实现了Deque接口,并且可调整大小的一个双向队列。ArrayDeque队列没有容量限制,它可以根据需要扩容。

ArrayDeque底层采用数组实现的。

ArrayDeque特性:

  • ArrayDeque是一个可扩容的双端队列。
  • 内部使用数组存储数据。
  • ArrayDeque不是线程安全的。
  • ArrayDeque禁止使用空元素。
  • ArrayDeque用作堆栈时,比Stack要快,当它用作队列时,比LinkedList要快。
  • ArrayDeque最大容量是2^30个元素。
  • ArrayDeque队列的容量必须满足size = 2的n次幂。
  • ArrayDeque有两个类属性,head和tail,两个指针,可以在数组任意一端进行数据插入或删除。

Android中的使用

我们在分析AsyncTask时,它的内部实现就用到了ArrayDeque作为一个先进先出的队列。

private static class SerialExecutor implements Executor {
        final ArrayDeque<Runnable> mTasks = new ArrayDeque<Runnable>();
        Runnable mActive;

        public synchronized void execute(final Runnable r) {
            mTasks.offer(new Runnable() {
                ……
            });
            ……
        }

        protected synchronized void scheduleNext() {
            if ((mActive = mTasks.poll()) != null) {
                THREAD_POOL_EXECUTOR.execute(mActive);
            }
        }
    }

这里使用了offer方法添加元素,使用poll方法获取元素。我们知道LinkedList也可以实现队列操作,那么为什么会使用ArrayDeque呢?它有哪些优势呢?

我们接下来分析它的实现原理。

ArrayDeque解析


首先我们来看ArrayDeque的构建。

ArrayDeque队列的容量必须满足2的n次幂,它是如何保证的呢?如果我们给它的构造函数传递一个非2次幂的值,它又是如何来调整大小的呢?

ArrayDeque的队列容量初始化原理

我们来看它的构造函数:
public ArrayDeque() {
        elements = new Object[16];
    }
    public ArrayDeque(int numElements) {
        allocateElements(numElements);
    }
    public ArrayDeque(Collection<? extends E> c) {
        allocateElements(c.size());
        addAll(c);
    }
ArrayDeque有3个重载构造方法:
  • 无参数时,ArrayDeque的初始大小是16,该值满足2的次幂要求。
  • 参数是名为numElements的int值,它允许使用者自定义容器的初始大小,该构造函数只是调用了allocateElements(numElements),看来大小调整的秘密就在allocateElements方法中了。
  • 参数为Collection的构造方法中,也同样调用了allocateElements方法,也侧面验证了allocateElements是实现ArrayDeque大小是2的次幂的关键。

那么我们就来分析allocateElements的实现吧。

allocateElements方法实现初始化容器原始大小

ArrayDeque的构造函数中会调用allocateElements方法,来实现队列的初始化。

allocateElements的源码:

private static final int MIN_INITIAL_CAPACITY = 8;
    private void  allocateElements(int numElements) {
        int initialCapacity = MIN_INITIAL_CAPACITY;
        // 如果队列容量size的大小,大于最小值时,需要通过位运算来设置成一个符合2的n次幂的值。例如,如果参数值是9~15之间的任意值,size(initialCapacity变量的值)的最终大小就是16,如果参数值是17~31之间的任意值,则size的最终大小是32。
        if (numElements >= initialCapacity) {
            initialCapacity = numElements;
            initialCapacity |= (initialCapacity >>>  1);
            initialCapacity |= (initialCapacity >>>  2);
            initialCapacity |= (initialCapacity >>>  4);
            initialCapacity |= (initialCapacity >>>  8);
            initialCapacity |= (initialCapacity >>> 16);
            initialCapacity++;

            if (initialCapacity < 0)    // 如果值太大溢出(即小于0),需要缩小2倍
                initialCapacity >>>= 1; // 最大运行值是 2^30
        }
        elements = new Object[initialCapacity];
    }
逻辑解析:
  • 这里首先判断队列容量值numElements是否小于最小初始值,如果小于,则直接以最小初始值为size,初始化队列容器,这里最小size是8。
  • 如果大于最小初始值,则进行一系列”位“操作。
  • 初始initialCapacity的二进制位中,高位往低位找,找到第一个”1“,经过这一系列位操作之后,会把”1“之后的所有位上的值,都变成”1“。例如,initialCapacity值等于9,则位操作之前是”1001“,位操作之后变为”1111“,也就是15。
  • 位操作之后,initialCapacity自增,上例中就由15变为16了,也就符合了2的次幂规则了,其他数值也可以用此方法推算出结果。
  • 最后,如果initialCapacity值溢出,则左移一位。

ArrayDeque的扩容原理

当两个双端指针相遇时,也就是head等于tail时,则表示数组需要扩容了。扩容是通过方法doubleCapacity来实现的。

doubleCapacity方法:
private void doubleCapacity() {
        assert head == tail; //断言
        int p = head; //临时变量p,等于head
        int n = elements.length;//扩容前的size
        int r = n - p; // head右侧元素个数
        int newCapacity = n << 1;//size扩大1倍
        if (newCapacity < 0)//溢出了,抛出错误
            throw new IllegalStateException("Sorry, deque too big");
        Object[] a = new Object[newCapacity];//创建新的数组,容量为原来的2倍
        System.arraycopy(elements, p, a, 0, r);//将head右侧数据从原来数组copy到新数组
        System.arraycopy(elements, 0, a, r, p);//将head左侧的数据从原数组copy到新数组
        // Android-added: Clear old array instance that's about to become eligible for GC.
        // This ensures that array elements can be eligible for garbage collection even
        // before the array itself is recognized as being eligible; the latter might
        // take a while in some GC implementations, if the array instance is longer lived
        // (its liveness rarely checked) than some of its contents.
        Arrays.fill(elements, null);//将原数组清空
        elements = a;//将扩容后的数组赋给elements
        head = 0;//设置head为0
        tail = n;//设置tail为扩容前的数组大小,作为指针起点
    }
逻辑解析:
  • 计算出head指针右侧元素个数,之后需要使用它来进行数据复制。
  • 数组容量扩大1倍,如果溢出(大于2^30),则报错。
  • 创建新的数组作为容器,然后原数组的内容copy到新数组。
  • 将原数组元素都置位null。
  • 将新数组赋给elements。
  • 设置head的值为0,tail的值为原数组的大小n。这时,新数组的[0~n)指针位均已存储元素。

下面我们再来分析ArrayDeque的几个关键方法。

队列头部添加元素addFirst

head代表了队列头当前插入点的指针,它的初始值是0。

在队列头部添加一个元素,使用addFirst方法:
public void addFirst(E e) {
        if (e == null)
            throw new NullPointerException();
        elements[head = (head - 1) & (elements.length - 1)] = e;
        if (head == tail)
            doubleCapacity();
    }
逻辑解析:

这里的关键点就在head的计算上,head - 1 的使head得指针向左移动一位。例如,当head初始值为0时,队列容量是8时,head - 1的值就是-1,用二进制表示是1111,(elements.length - 1)的二进制表示是0111,两者按位&操作,结果就是7。这种操作结果类似于取余操作,但使用起来非常方便。

也就是说addFirst的第一个元素,在数组中的位置是elements[elements.length - 1],之后每次插入操作,head指针都会向左移动。

当head指针和tail指针相等时,表示数组容器已经满了,需要进一步扩容了,扩容调用doubleCapacity方法即可。

队列尾添加元素addLast

head代表了队列尾当前插入点的指针,它的初始值是0。

在队列尾添加一个元素,使用addLast方法:
public void addLast(E e) {
        if (e == null)
            throw new NullPointerException();
        elements[tail] = e;
        if ( (tail = (tail + 1) & (elements.length - 1)) == head)
            doubleCapacity();
    }
逻辑解析:
  • 这里直接把元素存储到tail指针位置,初始时tail的值是0,也就是第一个在尾部添加的元素数组索引是0。
  • 接下来tail指针右移,指向新的位置。这里同样使用了位运算来实现,以队列容量大小8为例,当tail的位置在第八位时(tail值是7),tail的二进制是0111,加1后变为1000,按位&二进制0111(elements.length - 1的二进制),结果为0。
  • 同样,当tail等于head时,扩容。

队列头弹出元素pollFirst

public E pollFirst() {
        final Object[] elements = this.elements;
        final int h = head;
        @SuppressWarnings("unchecked")
        E result = (E) elements[h];
        // Element is null if deque empty
        if (result != null) {
            elements[h] = null; // Must null out slot
            head = (h + 1) & (elements.length - 1);
        }
        return result;
    }
逻辑解析:
  • 获取head指针位置的元素:elements[h],并赋值给result变量。
  • 当元素不为null时,将elements[h]置位null,并且head指针右移(同样使用了位操作)。
  • 最后返回result变量。

队列尾弹出元素pollLast

public E pollLast() {
        final Object[] elements = this.elements;
        final int t = (tail - 1) & (elements.length - 1);
        @SuppressWarnings("unchecked")
        E result = (E) elements[t];
        if (result != null) {
            elements[t] = null;
            tail = t;
        }
        return result;
    }
逻辑解析:
  • 设置临时变量t,并将tail左移一位之后赋值给t。
  • 获取队尾元素,并赋值给result。
  • 如果result不为null,将elements[t]置位null,并且将t赋值给tail指针。
  • 最后返回result变量。

offer方法和poll方法


前文原理解析中,介绍了ArrayDeque的主要方法,但并没有AsyncTask中使用的offer方法和poll方法。其实这两个方法都是基于之前方法实现的封装接口,我们来看。

offer方法

public boolean offer(E e) {
        return offerLast(e);
    }
    public boolean offerLast(E e) {
        addLast(e);
        return true;
    }

offer方法调用了offerLast方法,offerLast方法最终调用的是offerLast方法,offerLast方法之前已经做过分析了,它实现了在队尾添加元素。

poll方法

public E poll() {
        return pollFirst();
    }

poll方法直接return了pollFirst方法,pollFirst方法实现了队列头部弹出。

小结


我们在本文中分析了ArrayDeque的实现、使用、及原理,下面来简单总结一下:

  1. ArrayDeque是一个实现了Deque接口的双向队列。
  2. ArrayDeque的容量大小是可以动态调整的,并且容量大小必须满足是2的n次幂。
  3. ArrayDeque内部是使用数组来实现数据存储的。
  4. ArrayDeque不是线程安全的。
  5. ArrayDeque禁止使用空元素。
  6. ArrayDeque用作堆栈时,比Stack要快,当它用作队列时,比LinkedList要快。
  7. ArrayDeque最大容量是2^30个元素。
  8. ArrayDeque有两个类属性,head和tail,两个指针,可以在数组任意一端进行数据插入或删除。
  9. ArrayDeque在Android中有大量的应用。
  10. 在队列头部添加元素,可使用addFirst方法,在队列尾添加元素可使用addLast方法,队列头弹出元素可使用pollFirst方法,队列尾弹出元素可使用pollLast方法。
  11. ArrayDeque的offer方法可以实现在队尾添加元素,它内部是调用offerLast方法实现的。
  12. ArrayDeque的poll方法可以在队列头弹出元素,它的内部是调用pollFirst方法实现的。
  13. ArrayDeque中,大量使用了位操作,非常巧妙的省去了边界处理等问题,简化了操作。