ArraysLIst的动态扩容源码详解

对于ArraysList的添加,首先会对当前容量进行判断

 //添加操作
	public boolean add(E e) {
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        elementData[size++] = e;		//添加元素并把数组长度+1
        return true;
    }

	//确保内部容量,容量不足就执行扩容操作
    private void ensureCapacityInternal(int minCapacity) {
        ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
    }

创建ArraysList对象时,初始容量为0,当执行添加操作,就会判断到elementData为空,此时就创建一个容量为10的新数组,覆盖原来的elementData数组,初始化成功

    private static int calculateCapacity(Object[] elementData, int minCapacity) {
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {	//数组为空返回 DEFAULT_CAPACITY(10)
            return Math.max(DEFAULT_CAPACITY, minCapacity);
        }
        return minCapacity;	//当数组有数据时就直接返回+1后的长度
    }
    

对当前容量进行判断

  • 当添加时的size++不超过elementData数组的容量length,就不扩容
  • 当添加时的size++超过elementData数组的容量length,就会执行扩容
   private void ensureExplicitCapacity(int minCapacity) {
        modCount++;

        // overflow-conscious code,		length为数组当前容量 
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }

		//扩容
    private void grow(int minCapacity) {
        // overflow-conscious code
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        if (newCapacity - minCapacity < 0)	//初始化数组容量10
            newCapacity = minCapacity;
        if (newCapacity - MAX_ARRAY_SIZE > 0)	//扩容后数组容量 > MAX_ARRAY_SIZE(2的31次幂-8)时
            newCapacity = hugeCapacity(minCapacity);
        // minCapacity is usually close to size, so this is a win:
        elementData = Arrays.copyOf(elementData, newCapacity);
    }
   

扩容会有一个最大容量限制

	//限制最大扩容量为 Integer.MAX_VALUE
    private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }

根据扩容后容量值,把原数组数据复制到新创建的数组,扩容成功

	//执行Arrays数组的copyOf方法【】10
   public static <T> T[] copyOf(T[] original, int newLength) {
        return (T[]) copyOf(original, newLength, original.getClass());
    }

	//
    public static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {
        @SuppressWarnings("unchecked")
        T[] copy = ((Object)newType == (Object)Object[].class)
            ? (T[]) new Object[newLength]		//空数组时new一个容量为10的新数组
            : (T[]) Array.newInstance(newType.getComponentType(), newLength);	//new一个容量为扩容后的数组
        System.arraycopy(original, 0, copy, 0,
                         Math.min(original.length, newLength));	//把原数组放到新数组前面,保证新添加数据在原数组后面
        return copy;
    }

//Array.newInstance方法
class Array {
    public static Object newInstance(Class<?> componentType, int length)
        throws NegativeArraySizeException {
        return newArray(componentType, length);
    }
    
    private static native Object newArray(Class<?> componentType, int length)
        throws NegativeArraySizeException;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值