关于ArrayList初始化时,初始容量的问题
这里写自定义目录标题
首先来看ArrayList的构造方法有哪些
一、ArrayList的构造方法
无参构造
传入指定参数
这个暂时不用管
在ArrayList中有一个elementDate的数组,ArrayList底层就是用Object数组来实现的
这里还定义了一个空数组,在ArrayList使用无参构造方法时使用
回到问题,如果ArrayList在初始化时,没有传递参数,如:
ArrayList list=new ArrayList();//这里先不使用泛型
这时,会调用ArrayList的无参构造方法,也就是
这时把一个定义的Object的空数组的引用赋值给了当前对象的elementData数组,即在ArrayList在调用无参构造时,ArrayList是一个空数组,并没有指定默认容量。
二、在第一次添加元素时,会指定ArrayList的数组容量
首先我们查看ArrayList,add()方法的源码
这个有两个add()方法,第一个是add方法传递一个参数,可以直接放入元素。第二个add方法传递一个数组下标和要存放的元素。这里我们以第一个add方法讨论。
在存放第一个元素时,调用add方法,在add方法里第一行,这个方法表示进行数组是否需要扩容的判断,这里的size也是ArrayList里定义的一个变量,表示数组里的元素有多少个。也就是说我们在之前使用无参构造时,ArrayList是一个空数组,即size==0,在执行第一行方法时传递的是size+1,即传递参数为1,我们进入到这个ensureCapacitiInternal方法里。
这里首先看这个方法,传递了elementData数组,和我们之前传进来的参数1
再往下看这个calculateCapacity方法
这里就是第一次给ArrayList数组设置初始容量的关键地方,在if语句就判断了,这个数组是不是为一个空数组。我们之前在调用无参构造时就把elmentData设置为空数组。所以他会执行if语句里的语句,返回了一个Math.max(DEFAULT_CAPACITY,minCapacity),也就是返回他们两个之间的最大值,而DEFAULT_CAPACITY,在ArrayList源码里设置为10。
所以在和minCapacity为1的参数进行比较时,会返回10。所以我们回来到这里来看看。
然后传递参数10进入这个ensureExplicitCapacity方法中,我们进到这个方法的源码来看看。
这里我们直接看if语句,我们把刚刚传进来的参数10,减去这个数组的长度,即10-0,条件成立,进入到if语句中,然后执行这个grow方法,并且把10传递进去。再进入到grow方法里。
这个grow方法其实就是扩容数组容量的,首先,下面这两行代码
对原来的数组容量进行1.5扩容,即newCapacity就是扩容后的长度。
因为我们前面初始化的数组是没有长度的,所以这里扩容后还是0。
接下来这个if语句,用来和我们之前的传入的参数(也就是之前传递进来的参数10)进行比较,最后把10赋值给了newCapacity。
第二个if语句是判断数组容量有没有达到最大值,
这里源码里给出了最大值大小ArrayList private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8 = 2147483639
所以直接跳过这部分。最后,返回了容量为10新的的数组。
到这里,ArrayList的初始容量已经才变为10。
总结:
在new ArrayList()时括号里没有指定参数的话,则ArrayList这时的数组容量为0,只有在第一次添加元素时,才会把数组容量设为10。
以上是我对ArrayList初始化时,数组容量大小问题的理解,如果有错误的地方,希望大神们能够指正!