ArrayList 的底层是数组队列,相当于动态数组。与 Java 中的数组相比,它的容量能动态增长。在添加大量元素前, 应用程序可以使用ensureCapacity操作来增加 ArrayList 实例的容量。这可以减少递增式再分配的数量。
由于ArrayList使用的是数组存储结构,所有数组的特性都可以用于此ArrayList集合的操作。 ArrayList非线程安全,如果有多线程共同操作这个list实例,就需要将其加锁, 将锁加在包含这个list的对象上,或者使用如下形式:
List list = Collections.synchronizedList(new ArrayList(...));
扩容机制
add
如果不指定List的初始化大小,在初始化的时候会初始化一个空数组, 在添加第一个元素的时候才进行数组的初始化扩容,默认值为10;
// 将指定的元素追加到此列表的末尾。
public boolean add(E e) {
// 添加元素之前,先调用ensureCapacityInternal方法
ensureCapacityInternal(size + 1); // Increments modCount!!
// Increments modCount!!
elementData[size++] = e;
return true;
}
grow
当十个元素位置都被填满,会进行第二次扩容,此次扩容量为当前集合的一半, (扩容使用的是位移算法,10»1=5,15»1=7) 整个过程如下:
0 + 10 => 10 + 5 => 15 + 7 => 22 + 11 => 33 …
扩容代码具体实现如下:
private void grow(int minCapacity) {
// oldCapacity为旧容量,newCapacity为新容量
int oldCapacity = elementData.length;
// 将oldCapacity 右移一位,容量扩大到1.5倍。(位运算的速度远远快于整除运算)
int newCapacity = oldCapacity + (oldCapacity >> 1);
// 然后检查新容量是否大于最小需要容量,若还是小于最小需要容量,那么就把最小需要容量当作数组的新容量,
if (newCapacity - minCapacity < 0)
newCapacity = minCapacity;
// 如果新容量大于 MAX_ARRAY_SIZE,进入(执行) `hugeCapacity()` 方法来比较 minCapacity 和 MAX_ARRAY_SIZE
if (newCapacity - MAX_ARRAY_SIZE > 0)
newCapacity = hugeCapacity(minCapacity);
// minCapacity is usually close to size, so this is a win:
elementData = Arrays.copyOf(elementData, newCapacity);
}
所以当数组长度越大的时候,扩容消耗的资源越多; 如果能够在数组初始化的时候大概估算出数组长度上限进行初始化, 就能够减少数组扩容所引起的资源消耗,提升性能。
数组拷贝
阅读源码的话,我们就会发现 ArrayList 中大量调用了 System.arraycopy()
和 Arrays.copyOf()
这两个方法。比如:我们上面讲的扩容操作以及 add(int index, E element)、toArray() 等方法中都用到了该方法。
System.arraycopy()
/**
* @param src 源数组
* @param srcPos 源数组中的起始位置
* @param dest 目标数组
* @param destPos 目标数组中的起始位置
* @param length 要复制的数组元素的数量
*/
public static native void arraycopy(Object src, int srcPos,
Object dest, int destPos,
int length);
Arrays.copyOf()
看两者源代码可以发现 copyOf()内部实际调用了 System.arraycopy() 方法:
public static <T> T[] copyOf(T[] original, int newLength) {
return (T[]) copyOf(original, newLength, original.getClass());
}
public static <T,U> 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;
}
参考资料
文档信息
- 本文作者:Bob.Zhu
- 本文链接:https://adolphor.github.io/2016/08/11/JDK-ArrayList/
- 版权声明:自由转载-非商用-非衍生-保持署名(创意共享3.0许可证)