Java容器系列-ArrayList源码分析

ArrayList 是使用的最为普遍的一个容器。ArrayList 的类的继承层次图以下:java

ArrayList 实现了 CollectionList 接口,同时也实现了 CloneableRandomAccess,因此 ArrayList 能够被拷贝以及具备随机访问的特性。数组

本文基于 JDK1.8安全

成员变量

在 ArrayList 类的头部,定义了如下几个成员变量。微信

private static final int DEFAULT_CAPACITY = 10;
private static final Object[] EMPTY_ELEMENTDATA = {};
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
transient Object[] elementData;
private int size;
复制代码

这几个变量构成了 ArrayList 的基础。数据结构

DEFAULT_CAPACITY 表示 ArrayList 的初始容量。elementData 是存储具体数据的数组,也就是是说,ArrayList 底层数据结构就是一个数组,size 表示 ArrayList 中元素的个数。EMPTY_ELEMENTDATA 表示一个空的 ArrayList 对象,但 ArrayList 中没有数据时,elementData 指向的就是这个数组对象。DEFAULTCAPACITY_EMPTY_ELEMENTDATA 也表示空的 ArrayList,它只会在实例化一个不带参数的 ArrayList 的时候被使用一次:多线程

ArrayList list = new ArrayList(); // 此时 elementData 指向的就是 DEFAULTCAPACITY_EMPTY_ELEMENTDATA
复制代码

这些变量中须要注意的是 elementData 是不带访问修饰符的,这是为了让 ArrayList 的内部类能够方便的访问它,ArrayList 的内部类后面会讲到。elementData 变量是使用 transient 来修饰的,这表示在序列化的时候 elementData 是不会被序列化的,具体的序列化方式后面再讲。dom

初始化过程

在上面说到 DEFAULT_CAPACITY 是 ArrayList 的默认容量,值是10,可是须要注意的是,默认容量不必定用的上,在实例化 ArrayList 的时候分三种状况,第一种不给构造函数传参,可是此时会新建一个长度为 10 的对象数组。而是在添加第一个元素的时才会建立一个长度为 10 的数组,并把第一个元素添加进去。第二种状况会给构造参数传值 n,若是 n 大于0,那么就会直接建立一个长度为 n 的对象数组,若是 n 等于 0,那么就会把 EMPTY_ELEMENTDATA 赋值给 elementData。函数

第三种实例化特殊一点,是直接传入另外一个容器对象 c 来初始化 ArrayList 对象,此时会先检查 c 的长度,若是 c 容器里面没有元素,直接把 EMPTY_ELEMENTDATA 赋值给 elementData,若是 c 不为空,就会 c 中的元素拷贝到 elementData 中。post

扩容过程

扩容的过程能够用如下的流程图来表示:spa

扩容对 ArrayList 来讲是一个很重要的过程,这也是为何它比数组好用的缘由。

ArrayList 的扩容有两种方式,一个是自动扩容,一种是手动扩容。自动扩容每次会把当前容器的大小扩大 1.5 倍,手动扩容须要指定大小。既然已经有了自动扩容,那为何还须要手动扩容呢?设想一个场景,实例化一个 ArrayList 以后,你大概知道会填充一万个元素,若是这个时候自动扩容的话要通过屡次扩容才能装下这么多元素,可是手动指定容器大小的话只须要一次就能够了。

具体把 ArrayList 扩容到多大是由下面这段代码决定的:

private int newCapacity(int minCapacity) {
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        if (newCapacity - minCapacity <= 0) {
            if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA)
                return Math.max(DEFAULT_CAPACITY, minCapacity);
            if (minCapacity < 0) // overflow
                throw new OutOfMemoryError();
            return minCapacity;
        }
        return (newCapacity - MAX_ARRAY_SIZE <= 0)
            ? newCapacity
            : hugeCapacity(minCapacity);
    }

    private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE)
            ? Integer.MAX_VALUE
            : MAX_ARRAY_SIZE;
    }
复制代码

newCapacity 是根据当前的元素的个数计算出来的,右移一位表明除以2,因此 newCapacity 为当前容量的 1.5 倍。而后这个值会与传入的值 minCapacity 进行对比,两个值哪一个大就用哪一个。

为何每次自动扩容都能为当前大小的 1.5 倍呢?那是由于自动扩容的时候传入的 minCapacity 都只比当前的容量大 1,因此确定小于 newCapacity。而 newCapacity 就是 当前容量大小的 1.5 倍。

固然有一个状况例外,那就是若是在实例化 ArrayList 没有指定大小的话,ArrayList 会至少扩容到 10。这一机制是靠如下代码实现的:

if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA)
    return Math.max(DEFAULT_CAPACITY, minCapacity);
复制代码

扩容的时候,都是使用 Arrays.copyOf 将元素拷贝到新的容器中。因此基本都是 O(N) 的时间复杂度,代价很大。因此尽量减小扩容的次数。

注意:ArrayList 没有缩容的过程。

具体实现

ArrayList 中有了不少的方法,这些方法核心都是围绕 elementData 操做的。

siez()isEmpty() 方法想着简单,一个用来返回容器中的元素的数量,一个用来判断容器是否为空。

clone()toArray()toArray(T[] a) 这三个方法本质上都是对容器当前的元素作一个备份,都用到了 Arrays.copyOf() 方法。可是须要注意的是 toArray(T[] a) 的实现:

public <T> T[] toArray(T[] a) {
    if (a.length < size)
        // Make a new array of a's runtime type, but my contents:
        return (T[]) Arrays.copyOf(elementData, size, a.getClass());
    System.arraycopy(elementData, 0, a, 0, size);
    if (a.length > size)
        a[size] = null;
    return a;
}
复制代码

在这个方法中除了使用 Arrays.copyOf() 还用到了 System.arraycopy(),其实 Arrays.copyOf() 底层就是使用 System.arraycopy() 方法实现的。可是区别在于前者会返回一个一个新的数组,后者则是直接在原数组上进行操做。

ArrayList 中的 add()get()set()remove() 等方法用于元素的增删改查,实现并不复杂,只是在操做元素以前须要对容器的 size 进行检查,若是不知足操做要求,就会报出异常。

euqal() 类的方法主要都是对比每一个元素的类型、顺序和值是否一致。

在 JDK1.8 之后,出现了 removeIf() 方法,这个方法使得从容器中删除元素变得很简单。

迭代器

ArrayList 中有两个内部类 ItrListItr,主要方法以下:

private class Itr implements Iterator<E> {
    Itr() {}

    public boolean hasNext() {
    }

    public E next() {
      
    }

    public void remove() {
    }
}
复制代码
private class ListItr extends Itr implements ListIterator<E> {
    ListItr(int index) {
        super();
        cursor = index;
    }

    public boolean hasPrevious() {
    }

    public int nextIndex() {
    }

    public int previousIndex() {
    }

    public E previous() {
    }

    public void set(E e) {
    }

    public void add(E e) {
    }
}
复制代码

ListItr 继承了 Itr,这两个内部类都实现了迭代器模式,用于遍历 ArrayList 的元素。从上面的方法可知,Itr 和 ListItr 最大的区别在于 ListItr 能够从两个方向对容器的元素进行遍历。而 Itr 只能使用顺着一个方向进行遍历。

在 JDK1.8 之后,ArrayList 中有一个 ArrayListSpliterator 内部类,这个类用于分割容器。用于提高多线程环境中的处理效率:

ArrayList<Integer> list = new ArrayList<>();
for (int i = 0; i < 10; i++) {
    list.add(i);
}
Spliterator<Integer> s0 = list.spliterator();
System.out.println(s0.estimateSize()); // 5
Spliterator<Integer> s1 = s0.trySplit();
System.out.println(s1.estimateSize()); // 5
复制代码

s0 中有 10 个元素,在调用 s0.trySplit() 方法以后,s0 和 s1 中各有 5 个元素。而后能够对分割开的元素进行处理:

s0.forEachRemaining(i -> System.out.println(i));
复制代码

SubList

ArrayList 中还有一个内部类 SubList。SubList 用于返回 ArrayList 的一部分元素,内部的操做方法与 ArrayList 基本一致,可是须要注意的是,对 SubList 的操做会直接影响到原 ArrarList。

fail-fast 机制

在 ArrayList 中,checkForComodification()ConcurrentModificationException() 使用的频率很高。这个和 fail-fast 机制有关。

ArrayList 不是线程安全的,因此在对容器操做的过程当中,容器的元素倍其余的操做或者线程修改以后,就会出现 ConcurrentModificationException 异常。checkForComodification() 方法就是用来检查元素是否被修改。这个机制就称之为 fail-fast

后续会有其余的文章来介绍 fail-fast

(完)

原文

相关文章

关注微信公众号,聊点其余的

相关文章
相关标签/搜索