一 概述
深入理解Java集合中的源代码,可以帮助我们更好地了解大佬的意图,规避不必要的bug。
源码中的一段注释,提取关键信息
Resizable-array implementation of the List interface. Implements all optional list operations, and permits all elements, including null. In addition to implementing the List interface, this class provides methods to manipulate the size of the array that is used internally to store the list. (This class is roughly equivalent to Vector, except that it is unsynchronized.)
由上文注释可以大概得知:ArrayList是一个动态数组,实现了List接口以及list相关的所有方法,它允许所有元素的插入,包括null。另外,ArrayList和Vector除了线程不同步之外,大致相等。
二 属性
1 | //默认容量的大小 |
ArrayList 的属性非常少,就只有这些。其中最重要的莫过于 elementData 了,ArrayList 所有的方法都是建立在 elementData 之上。
三 方法
1 构造方法
1 | public ArrayList(int initialCapacity) { |
从构造方法中可以看出,默认情况下,elementData 是一个大小为 0 的空数组,当我们指定了初始大小的时候,elementData 的初始大小就变成了我们所指定的初始大小了。
2 get方法
1 | public E get(int index) { |
因为 ArrayList 是采用数组结构来存储的,所以它的 get 方法非常简单,先是判断一下有没有越界,之后就可以直接通过数组下标来获取元素了,所以 get 的时间复杂 度是 O(1)。
3 add方法
1 | public boolean add(E e) { |
ArrayList 的 add 方法也很好理解,在插入元素之前,它会先检查是否需要扩容,然 后再把元素添加到数组中最后一个元素的后面。在 ensureCapacityInternal 方法中, 可以看见,如果当 elementData 为空数组时,它会使用默认的大小去扩容。所以 说,通过无参构造方法来创建 ArrayList 时,它的大小其实是为 0 的,只有在使用到 的时候,才会通过 grow 方法去创建一个大小为 10 的数组。
第一个 add 方法的复杂度为 O(1),虽然有时候会涉及到扩容的操作,但是扩容的次 数是非常少的,所以这一部分的时间可以忽略不计。如果使用的是带指定下标的 add 方法,则复杂度为 O(n),因为涉及到对数组中元素的移动,这一操作是非常耗时的。
4 set方法
1 | public E set(int index, E element) { |
set 方法的作用是把下标为 index 的元素替换成 element,跟 get 非常类似,时间复杂度度为 O(1)。
5 remove方法
1 | public E remove(int index) { |
remove 方法与 add 带指定下标的方法非常类似,也是调用系统的 arraycopy 方法来 移动元素,时间复杂度为 O(n)。
6 grow方法
1 | private void grow(int minCapacity) { |
grow 方法是在数组进行扩容的时候用到的,从中可以看见,ArrayList 每次扩容 都是扩 1.5 倍,然后调用 Arrays 类的 copyOf 方法,把元素重新拷贝到一个新的数组 中去。
7 ize方法
1 | public int size() { |
size 方法非常简单,它是直接返回 size 的值,也就是数组中元素的个数间 复杂度为 O(1)。这里要注意一下,返回的并不是数组的实际大小。
8 indexOf & lastIndexOf
1 | public int indexOf(Object o) { |
indexOf 方法的作用是返回第一个等于给定元素的值的下标。它是通过遍历比较数组 中每个元素的值来查找的,所以它的时间复杂度是 O(n)。
lastIndexOf 的原理跟 indexOf 一样,而它仅仅是从后往前找起罢了。
四 Vector
很多方法都跟 ArrayList 一样,只是多加了个 synchronized 来保证线程安全,主要汇总二者不同点
Vector 比 ArrayList 多了一个属性:
1 | protected int capacityIncrement; |
这个属性是在扩容的时候用到的,它表示每次扩容只扩 capacityIncrement 个空间就 足够了。该属性可以通过构造方法给它赋值。先来看一下构造方法:
1 | public Vector(int initialCapacity, int capacityIncrement) { |
从构造方法中,可以看出 Vector 的默认大小也是 10,而且它在初始化的时候就 已经创建了数组了,这点跟 ArrayList 不一样。再来看一下 grow 方法
1 | private void grow(int minCapacity) { |
从 grow 方法中可以发现,newCapacity 默认情况下是两倍的 oldCapacity,而当 指定了 capacityIncrement 的值之后,newCapacity 变成了 oldCapacity+capacityIncrement。
五 总结
1、ArrayList 创建时的大小为 0;当加入第一个元素时,进行第一次扩容时,默认容量大小为10
2、ArrayList 每次扩容都以当前数组大小的 1.5 倍去扩容。 3、Vector 创建时的默认大小为 10
4、Vector 每次扩容都以当前数组大小的 2 倍去扩容。当指定了 capacityIncrement 之 后,每次扩容仅在原先基础上增加 capacityIncrement 个单位空间。
5、ArrayList 和 Vector 的 add、get、size 方法的复杂度都为 O(1),remove 方法的复 杂度为 O(n)。
6、ArrayList 是非线程安全的,Vector 是线程安全的。