深入理解ArrayList的扩容机制的原理

前言:

  在面试中我们经常被问到说ArrayList和数组有什么不同,但我认为我们有必要去深入了解下ArrayList的原理。我们在日常开发的过程中,ArrayList也是经常使用的,接下来和小编一起淡来学习下ArrayList原理吧。

下面的代码是基于JDK1.8的

    // 默认为 0    private int size;     /**     * Appends the specified element to the end of this list.     * @param e element to be appended to this list     * @return <tt>true</tt> (as specified by {@link Collection#add})     */    public boolean add(E e) {        // size[0] + 1 当作参数传入方法         ensureCapacityInternal(size + 1);        elementData[size++] = e;        return true;    }

 

现在,让我们来看一下ensurecapacitanyin()方法

这个方法实际上判断数组是否已经初始化,如果未初始化的话就赋值为10

    // 用于反序列化的transient    transient Object[] elementData;      // 定义空数组    private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};     // 给常量DEFAULT_CAPACITY赋值默认容量为10    private static final int DEFAULT_CAPACITY = 10;     /**    * minCapacity:存放值的下标 + 1(size + 1)     */    private void ensureCapacityInternal(int minCapacity) {        // 判断elementData是不是默认的空数组        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {            // 取得两个参数中的最大值:DEFAULT_CAPACITY --> 10            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);        }        // minCapacity = 10        ensureExplicitCapacity(minCapacity);    }

然后我们来剖析下ensureExplicitCapacity()方法

扩容无外乎下面两种方法:

1 .判断该集合是否达到扩容的标准

2 .如果扩容后的长度小于当前容量,则扩容的容量=当前容量(minCapaccity)

    private void ensureExplicitCapacity(int minCapacity) {        //记录变更的次数与线程的安全性有关        modCount++;        // 确定是否满足扩展要求:当前容量 > 数组长度        if (minCapacity - elementData.length > 0)            grow(minCapacity);    }     private void grow(int minCapacity) {        // 赋值之前的容量大小        int oldCapacity = elementData.length;        // 新的容量大小为:之前容量的大小 + (之前容量的大小 / 2) 注:“>>”的意思为除以2的1次方        int newCapacity = oldCapacity + (oldCapacity >> 1);         // 扩容后的容量小于前容量        if (newCapacity - minCapacity < 0)            newCapacity = minCapacity;         // 扩容后的容量大于arraylist最大容量时        if (newCapacity - MAX_ARRAY_SIZE > 0)            newCapacity = hugeCapacity(minCapacity);
        elementData = Arrays.copyOf(elementData, newCapacity);    }

 

这里我有两个问题,如下:

1、什么ArrayList的最大数组大小为Integer.MAX_VALUE-8?

2、为什么Integer.MAX_VALUE以0x7fffffff表示?

针对这两个问题,我查阅了相关资料得出以下结论

针对第一个问题的原因很简单:因为数组需要8 bytes去存储它自己的大小(2,147,483,648)(2^31)

下面为源码的注释:

/** * The maximum size of array to allocate. * Some VMs reserve some header words in an array. * Attempts to allocate larger arrays may result in * OutOfMemoryError: Requested array size exceeds VM limit */private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

翻译过来大概的意思是“要分配的数组的最大大小。一些vm在数组中保留一些头字。尝试分配较大的数组可能会导致*OutOfMemory错误:请求的数组大小超过了虚拟机限制“

针对第二个问题的原因是ox是16进制的7fffffff是16进制的值,是为了计算简单,计算机能够理解,但是我们不一定能够理解。 

接下来我会告诉你一些你可以面试到的实际情况, 一般的面试官只会问你基本的ArrayList概念,举两个栗子:

1 .例如ArrayList的基本实现是一个数组,数组的特征是插入删除慢,查询快,简单的说如果数组删除元素,需要向前移动所有元素,但查询仅直接依赖下标。

Arrays.copyOf(原数组,新容量)

我们来看看数组删除元素是如何操作的

例如,现在有一个数组:1、2、3、4。我删去了2,后面3、4都往前移一位

2.arrayList为什么线程不安全? 

很显然,没有使用关键字去加锁,扩容大小是原始的0.5倍(oldSize    oldSize/2)=newSize。

既然ArrayList是线程不安全的,但如果需要在多线程中使用,可以采用list<Object> list =Collections.synchronizedList(new ArrayList<Object>)来创建一个ArrayList对象。

原创声明:本文为【Java学习提升】原创博文,转载请注明出处。

本文来源于公众号:【Java学习提升】 专注于Java领域技术分享,Java知识体系学习、分享面试经验,让我们结伴而行,共同成长!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值