ArrayList 在我们日常开发中用到的非常多,我们知道ArrayList 内部是通过数组实现的,而数组的长度一经定义,就无法更改了。
那么问题就来了,ArrayList是如何实现扩容的呢?
ArrayList 的成员变量
1 | /** |
问题二:ArrayList 源码中为何定义两个 Object[] 呢?它们各有什么用处?
ArrayList 的构造方法
1、ArrayList 无参构造方法
注释上说,构造一个初始容量为10的空列表。实际上,Java8中使用了延迟初始化,使用无参构造方法,并不会马上创建长度为 10 的数组,而是在调用 add 方法添加第一个元素的时候才对 elementData 数组进行初始化(后面会看到)。
1 | /** |
2、指定初始容量的构造方法
传入初始容量,如果初始容量大于 0,那么直接创建一个指定大小的 Object 数组;如果等于0
1 | /** |
3、
1 | /** |
从上面文档中可以看出,我们可以通过 3 种方式创建 ArrayList 实例
add 元素
1 | /** |
为了确保 ArrayList 内部数组容量,add 方法首先调用 ensureCapacityInternal 方法,入参 minCapacity 为 ArrayList 包含的实际元素个数 size + 1。
1 | private void ensureCapacityInternal(int minCapacity) { |
计算容量,如果 ArrayList 是通过无参构造方法进行创建的,那么满足下面 if 条件,如果是添加第一个元素,则minCapacity 为 1,则数组扩容到 DEFAULT_CAPACITY 大小为10,这也对应了无参构造方法的注释 Constructs an empty list with an initial capacity of ten 。
1 | private static int calculateCapacity(Object[] elementData, int minCapacity) { |
1 | private void ensureExplicitCapacity(int minCapacity) { |
扩容
1 | /** |
扩容计算,int newCapacity = oldCapacity + (oldCapacity >> 1); oldCapacity 是ArrayList 内部数组长度,oldCapacity >> 1 是位运算的右移操作,右移一位相当于除以2,新的容量为之前容量的1.5倍
elementData = Arrays.copyOf(elementData, newCapacity); 对 elementData 数组进行扩容。
1 | private static int hugeCapacity(int minCapacity) { |
ArrayList 扩容每次都是原容量的1.5倍吗
https://blog.csdn.net/u014082714/article/details/85003562