JAVA中主要使用的List类有ArrayList和LinkedList,以及不常用的Vector。ArrayList是普通Array数组的扩充,基于队列实现,LinkedList则是基于链表实现。以上两者皆为线程非安全。Vector也是基于队列实现的,但不同于ArrayList的是,它是线程能安全的,因此资源开销相较于ArrayList也会更大。ArrayList相比于普通Array数组最大的特点就是可以动态扩充,下面介绍一下ArrayList的动态扩容功能。
一 初始化
首先有三种方式来初始化:
public ArrayList();
默认的构造器,将会以默认的大小来初始化内部的数组
public ArrayList(Collection<? extends E> c)
用一个ICollection对象来构造,并将该集合的元素添加到ArrayList
public ArrayList(int initialCapacity)
用指定的大小来初始化内部的数组
使用默认构造器的构造的ArrayList的长度是默认值,这个值在jdk1.6之前是10,之后是0。
二 扩容检查
扩容只会在容量不够时进行扩容,在扩容之前首先要进行容量检查。
// ① 是如何判断和扩容的。
private void ensureCapacityInternal(int minCapacity) {
//如果实际存储数组 是空数组,则最小需要容量就是默认容量
if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
}
ensureExplicitCapacity(minCapacity);
}
private void ensureExplicitCapacity(int minCapacity) {
modCount++;
//如果数组(elementData)的长度小于最小需要的容量(minCapacity)就扩容
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
/**
* Default initial capacity.
*/
private static final int DEFAULT_CAPACITY = 10;
使用ensureCapacityInternal方法来进行判断是否需要扩容。
三 扩容
最后的扩容步骤非常简单。
/*
*增加容量,以确保它至少能容纳
*由最小容量参数指定的元素数。
* @param mincapacity所需的最小容量
*/
private void grow(int minCapacity) {
// overflow-conscious code
int oldCapacity = elementData.length;
//>>位运算,右移动一位。 整体相当于newCapacity =oldCapacity + 0.5 * oldCapacity
// jdk1.7采用位运算比以前的计算方式更快
int newCapacity = oldCapacity + (oldCapacity >> 1);
if (newCapacity - minCapacity < 0)
newCapacity = minCapacity;
//jdk1.7这里增加了对元素个数的最大个数判断,jdk1.7以前是没有最大值判断的,MAX_ARRAY_SIZE 为int最大值减去8(不清楚为什么用这个值做比较)
if (newCapacity - MAX_ARRAY_SIZE > 0)
newCapacity = hugeCapacity(minCapacity);
// 最重要的复制元素方法
elementData = Arrays.copyOf(elementData, newCapacity);
}
首先扩大ArrayList的容量,扩展为原来容量1.5倍,然后调用Array的copyOf方法将原数组的内容直接拷贝过去。
以上就是List类的常用工具和其特性。