ArrayList是如何扩容的?


首先有三种构造方法,空参,指定大小,和指定集合

public ArrayList();

public ArrayList(Collection<? extends E> c)

public ArrayList(int initialCapacity)

后两种通过创造对象,或指定大小来初始化内部数据即可。

空参的时候,默认数组长度是0,也就是初始化数组长度为0

添加数据后,


public boolean add(E e) { //确保内部容量(通过判断,如果够则不进行操作;容量不够就扩容来确保内部容量)

 

     ensureCapacityInternal(size + 1); // ①Increments modCount!!

 

     elementData[size++] = e;//②

 

     return true;

 

 }

 

① ensureCapacityInternal方法名的英文大致是“确保内部容量”,size表示的是执行添加之前的元素个数,并非ArrayList的容量,容量应该是数组elementData的长度。ensureCapacityInternal该方法通过将现有的元素个数数组的容量比较。看如果需要扩容,则扩容。

②是将要添加的元素放置到相应的数组中。

根据传入的最小需要容量minCapacity来和数组的容量长度对比,若minCapactity大于或等于数组容量,则需要进行扩容。

综上所述,ArrayList的空参构造方法相当于在没指定initialCapacity时就是会使用延迟分配对象数组空间,当第一次插入元素时才分配10(默认)个对象空间。假如有20个数据需要添加,那么会分别在第一次的时候,将ArrayList的容量变为10 ;之后扩容会按照1.5倍增长。也就是当添加第11个数据的时候,Arraylist继续扩容变为10*1.5=15当添加第16个数据时,继续扩容变为15 * 1.5 =22个。

 

如果通过无参构造的话,初始数组容量为0,当真正对数组进行添加时,才真正分配容量。每次按照1.5倍(位运算)的比率通过copeOf的方式扩容。


/*
 *增加容量,以确保它至少能容纳
 *由最小容量参数指定的元素数。
 * @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);
 }


免责声明!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系本站邮箱yoyou2525@163.com删除。



 
粤ICP备18138465号  © 2018-2025 CODEPRJ.COM