ArrayList 是 Java 集合框架中 List接口 的一个实现类。
可以说 ArrayList 是我们使用最多的 List 集合,它有以下特点:
- 容量不固定,想放多少放多少(当然有最大阈值,但一般达不到)
- 有序的(元素输出顺序与输入顺序一致)
- 元素可以为 null
- 效率高
- size(), isEmpty(), get(), set() iterator(), ListIterator() 方法的时间复杂度都是 O(1)
- add() 添加操作的时间复杂度平均为 O(n)
- 其他所有操作的时间复杂度几乎都是 O(n)
- 占用空间更小
- 对比 LinkedList,不用占用额外空间维护链表结构
transient Object[] elementData
private static final int DEFAULT_CAPACITY = 10;
3.默认的空数组:
- private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
-
- private static final Object[] EMPTY_ELEMENTDATA = {};
private int size;
size <= capacity 。元素个数必定小于数组容量
5.数组最大容量:
private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
Integer.MAX_VALUE = 0x7fffffff
换算成二进制: 2^31 - 1,1111111111111111111111111111111
十进制就是 :2147483647,二十一亿多。
一些虚拟器需要在数组前加个 头标签,所以减去 8 。
当想要分配比 MAX_ARRAY_SIZE 大的个数就会报 OutOfMemoryError。
ArrayList 有三种构造函数:
1.无参构造
初始为空数组
-
- public ArrayList() {
-
- this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
- }
当调用无参构造方法时,elementData数组会被赋于DEFAULTCAPACITY_EMPTY_ELEMENTDATA,且在添加第一位元素时,elementData数组的容量被扩容为10。
2.有参构造
根据指定容量,创建个对象数组
-
- public ArrayList(int initialCapacity) {
- if (initialCapacity > 0) {
- this.elementData = new Object[initialCapacity];
- } else if (initialCapacity == 0) {
- this.elementData = EMPTY_ELEMENTDATA;
- } else {
- throw new IllegalArgumentException("Illegal Capacity: "+
- initialCapacity);
- }
- }
构造一个包含指定集合的元素列表
-
- public ArrayList(Collection extends E> c) {
- elementData = c.toArray();
- if ((size = elementData.length) != 0) {
- // c.toArray 有可能不返回一个 Object 数组
- if (elementData.getClass() != Object[].class)
- //使用 Arrays.copy 方法拷创建一个 Object 数组
- elementData = Arrays.copyOf(elementData, size, Object[].class);
- } else {
- // replace with empty array.
- this.elementData = EMPTY_ELEMENTDATA;
- }
- }
调用有参构造方法时,如果给定初始容量为0,或者传入集合为空集合(不是null),那么,将空数组EMPTY_ELEMENTDATA赋给elementData,此时在添加第一位元素时,不会扩容为10。
- public boolean add(E e) {
- //对数组的容量进行调整
- ensureCapacityInternal(size + 1); // Increments modCount!! modCount+1
- //如果elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA
- //在进行第一次元素增加后,列表将被扩展为DEFAULT_CAPACITY=10
- //size=1,elementData.length=10
- elementData[size++] = e;
- return true;
- }
2.add(int index, E element)
- //在指定位置添加一个元素
- public void add(int index, E element) {
- rangeCheckForAdd(index);
-
- //对数组的容量进行调整
- ensureCapacityInternal(size + 1); // Increments modCount!!
- //整体后移一位,效率不太好啊
- System.arraycopy(elementData, index, elementData, index + 1,
- size - index);
- elementData[index] = element;
- size++;
- }
3.addAll(Collection extends E> c)
在原来的集合的末尾,添加一个集合
-
- public boolean addAll(Collection extends E> c) {
- //把该集合转为对象数组
- Object[] a = c.toArray();
- int numNew = a.length;
- //增加容量
- ensureCapacityInternal(size + numNew); // Increments modCount
- //挨个向后迁移
- System.arraycopy(a, 0, elementData, size, numNew);
- size += numNew;
- //新数组有元素,就返回 true
- return numNew != 0;
- }
在指定位置,添加一个集合
-
- public boolean addAll(int index, Collection extends E> c) {
- rangeCheckForAdd(index);
-
- Object[] a = c.toArray();
- int numNew = a.length;
- ensureCapacityInternal(size + numNew); // Increments modCount
-
- int numMoved = size - index;
- //原来的数组挨个向后迁移
- if (numMoved > 0)
- System.arraycopy(elementData, index, elementData, index + numNew,
- numMoved);
- //把新的集合数组 添加到指定位置
- System.arraycopy(a, 0, elementData, index, numNew);
- size += numNew;
- return numNew != 0;
- }
- private void ensureCapacityInternal(int minCapacity) {
- //还没有添加元素,即elementData=={}
- if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
- //最小容量取默认容量和 当前元素个数 最大值
- minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
- }
-
- ensureExplicitCapacity(minCapacity);
- }
- private void ensureExplicitCapacity(int minCapacity) {
- modCount++;
-
- // 容量不够了,需要扩容
- if (minCapacity - elementData.length > 0)
- grow(minCapacity);
- }
- public void ensureCapacity(int minCapacity) {
- int minExpand = (elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA)
- // 不是默认的数组,说明已经添加了元素
- ? 0
- // 默认的容量
- : DEFAULT_CAPACITY;
-
- if (minCapacity > minExpand) {
- //当前元素个数比默认容量大
- ensureExplicitCapacity(minCapacity);
- }
- }
我们可以主动调用 ensureCapcity 来增加 ArrayList 对象的容量,这样就避免添加元素满了时扩容、挨个复制后移等消耗。
- private void grow(int minCapacity) {
- int oldCapacity = elementData.length; //elementData数组的容量,不是元素数量
- // 1.5 倍 原来容量
- int newCapacity = oldCapacity + (oldCapacity >> 1);//右移一位
-
- //如果当前容量还没达到 1.5 倍旧容量,就使用当前容量,省的站那么多地方
- if (newCapacity - minCapacity < 0)
- newCapacity = minCapacity;
-
- //新的容量居然超出了 MAX_ARRAY_SIZE
- if (newCapacity - MAX_ARRAY_SIZE > 0)
- //最大容量可以是 Integer.MAX_VALUE
- newCapacity = hugeCapacity(minCapacity);
-
- // minCapacity 一般跟元素个数 size 很接近,所以新建的数组容量为 newCapacity 更宽松些
- elementData = Arrays.copyOf(elementData, newCapacity);
- }
minCapacity一般等于元素个数或者默认容量10
- private static int hugeCapacity(int minCapacity) {
- if (minCapacity < 0) // overflow
- throw new OutOfMemoryError();
- //判断元素个数是否大于最大容量
- return (minCapacity > MAX_ARRAY_SIZE) ? //MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
- Integer.MAX_VALUE :
- MAX_ARRAY_SIZE;
- }
- E elementData(int index) {
- return (E) elementData[index];
- }
- //获取
- public E get(int index) {
- //判断index是否在范围内
- rangeCheck(index);
- //直接根据数组角标返回元素,快的一比
- return elementData(index);
- }
将列表中指定位置的元素替换为指定的元素。
- //修改
- public E set(int index, E element) {
- rangeCheck(index);
- E oldValue = elementData(index);
-
- //直接对数组操作
- elementData[index] = element;
- //返回原来的值
- return oldValue;
- }
- //根据位置删除
- public E remove(int index) {
- rangeCheck(index);
-
- modCount++;
- E oldValue = elementData(index);
-
- //numMoved等于数组元素-索引位置-1.如果numMoved等于0。说明删除的元素是最后一位,
- //然后跳过if语句。
- int numMoved = size - index - 1;
- if (numMoved > 0)
- System.arraycopy(elementData, index+1, elementData, index,
- numMoved);
- //原数组中最后一个元素删除,数组容量不变
- elementData[--size] = null; // clear to let GC do its work
-
- return oldValue;
- }
- public boolean remove(Object o) {
- if (o == null) {
- //挨个遍历找到目标
- for (int index = 0; index < size; index++)
- if (elementData[index] == null) {
- //快速删除
- fastRemove(index);
- return true;
- }
- } else {
- for (int index = 0; index < size; index++)
- if (o.equals(elementData[index])) {
- fastRemove(index);
- return true;
- }
- }
- return false;
- }
- private void fastRemove(int index) {
- modCount++;
- int numMoved = size - index - 1;
- if (numMoved > 0)
- System.arraycopy(elementData, index+1, elementData, index,
- numMoved);
- elementData[--size] = null; // clear to let GC do its work
- }
- public boolean retainAll(Collection> c) {
- Objects.requireNonNull(c);
- return batchRemove(c, true);
- }
- private boolean batchRemove(Collection> c, boolean complement) {
- final Object[] elementData = this.elementData;
- //使用两个变量,一个负责向后扫描,一个从 0 开始,等待覆盖操作
- int r = 0, w = 0;
- boolean modified = false;
- try {
- //遍历 ArrayList 集合
- for (; r < size; r++)
- //如果指定集合中是否有这个元素,根据 complement 判断是否往前覆盖删除
- if (c.contains(elementData[r]) == complement)
- elementData[w++] = elementData[r];
- } finally {
- //发生了异常,直接把 r 后面的复制到 w 后面
- if (r != size) {
- System.arraycopy(elementData, r,
- elementData, w,
- size - r);
- w += size - r;
- }
- if (w != size) {
- // 清除多余的元素,clear to let GC do its work
- for (int i = w; i < size; i++)
- elementData[i] = null;
- modCount += size - w;
- size = w;
- modified = true;
- }
- }
- return modified;
- }
-
- public void clear() {
- modCount++;
- //并没有直接使数组指向 null,而是逐个把元素置为空
- //下次使用时就不用重新 new 了
- for (int i = 0; i < size; i++)
- elementData[i] = null;
-
- size = 0;
- }
- public boolean contains(Object o) {
- return indexOf(o) >= 0;
- }
- public int indexOf(Object o) {
- if (o == null) {
- for (int i = 0; i < size; i++)
- if (elementData[i]==null)
- return i;
- } else {
- for (int i = 0; i < size; i++)
- if (o.equals(elementData[i]))
- return i;
- }
- return -1;
- }
- public int lastIndexOf(Object o) {
- if (o == null) {
- for (int i = size-1; i >= 0; i--)
- if (elementData[i]==null)
- return i;
- } else {
- for (int i = size-1; i >= 0; i--)
- if (o.equals(elementData[i]))
- return i;
- }
- return -1;
- }
- public Object[] toArray() {
- return Arrays.copyOf(elementData, size);
- }
- public
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());
- //全部元素拷贝到 数组 a
- System.arraycopy(elementData, 0, a, 0, size);
- if (a.length > size)
- a[size] = null;
- return a;
- }
3. Arrays.copyOf()
- public static
T[] copyOf(U[] original, int newLength, Class extends T[]> newType) { - @SuppressWarnings("unchecked")
- T[] copy = ((Object)newType == (Object)Object[].class)
- ? (T[]) new Object[newLength]
- : (T[]) Array.newInstance(newType.getComponentType(), newLength);
- System.arraycopy(original, 0, copy, 0,
- Math.min(original.length, newLength));
- return copy;
- }
如果 newType 是一个对象对组,就直接把 original 的元素拷贝到 对象数组中;
否则新建一个 newType 类型的数组。
1.rangeCheck
- private void rangeCheck(int index) {
- if (index >= size)
- throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
- }
检查给定索引是否在范围内,它总是在数组访问之前立即使用,如果索引为负数,则抛出ArrayIndexOutOfBoundsException异常。
2.rangeCheckForAdd
- private void rangeCheckForAdd(int index) {
- if (index > size || index < 0)
- throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
- }
一个由add和addAll使用的rangeCheck版本。
3.outOfBoundsMsg
- private String outOfBoundsMsg(int index) {
- return "Index: "+index+", Size: "+size;
- }
在 Java 集合深入理解:Abstractist 中我们介绍了 RandomAccess,里面提到,支持 RandomAccess 的对象,遍历时使用 get 比 迭代器更快。
由于 ArrayList 继承自 RandomAccess, 而且它的迭代器都是基于 ArrayList 的方法和数组直接操作,所以遍历时 get 的效率要 >= 迭代器。
- int i=0, n=list.size(); i < n; i++)
- list.get(i);
比用迭代器更快:
- for (Iterator i=list.iterator(); i.hasNext(); )
- i.next();
另外,由于 ArrayList 不是同步的,所以在并发访问时,如果在迭代的同时有其他线程修改了 ArrayList, fail-fast 的迭代器 Iterator/ListIterator 会报 ConcurrentModificationException 错。
因此我们在并发环境下需要外部给 ArrayList 加个同步锁,或者直接在初始化时用 Collections.synchronizedList 方法进行包装:
List list = Collections.synchronizedList(new ArrayList(...));