`

探索ArrayList自动改变size真相

 
阅读更多

探索ArrayList自动改变size真相

ArrayList的列表对象实质上是存储在一个引用型数组里的,有人认为该数组有“自动增长机制”可以自动改变size大小。正式地说,该数组是无法改变

大小的,实际上它只是改变了该引用型数组的指向而已。下面,让我们来看看java是怎样实现ArrayList类的。

一、ArrayList类的实质

     ArrayList底层采用Object类型的数组实现,当使用不带参数的构造方法生成ArrayList对象时,
实际上会在底层生成一个长度为10的Object类型数组。

    首先,ArrayList定义了一个私有的未被序列化的数组elementData,用来存储ArrayList的对象列表(注意只定义未初始):
  private transient Object[] elementData;
  
    其次,以指定初始容量(Capacity)或把指定的Collection转换为引用型数组后实例化elementData数组;如果没有指定,则预置初始容量为10进行

实例化。把私有数组预先实例化,然后通过copyOf方法覆盖原数组,是实现自动改变ArrayList的大小(size)的关键。有人说ArrayList是复杂的数组,我

认为不如说ArrayList是关于数组的系统的方法组合。

  ArrayList的构造方法源码如下:

    // 用指定的初始容量构造一个空列表。

    public ArrayList(int initialCapacity) {
        super();
        if (initialCapacity < 0)
            throw new IllegalArgumentException("Illegal Capacity: "+initialCapacity);
        this.elementData = new Object[initialCapacity];//属性指向新建长度为初始容量的临时数组
    }

    // 使用初始容量10构造一个空列表
    public ArrayList() {
        this(10);
    }

    / *构造包含利用collection的迭代器按顺序返回的指定collection元素的列表
     * @param c 集合,它的元素被用来放入列表t
     * @throws NullPointerException 如果指定集合为 null
     */
    public ArrayList(Collection<? extends E> c) {
        elementData = c.toArray();//用Collection初始化数组elementData
        size = elementData.length;
        if (elementData.getClass() != Object[].class)
            elementData = Arrays.copyOf(elementData, size, Object[].class);
    }

二、ArrayList实现自动改变size机制

   为了实现这一机制,java引进了Capacity和size概念,以区别数组的length。为了保证用户增加新的列表对象,java设置了最小容量(minCapacity)

,通常情况上,它大于列表对象的数目,所以Capactiy虽然就是底层数组的长度(length),但是对于最终用户来讲,它是无意义的。而size存储着列表

对象的数量,才是最终用户所需要的。为了防止用户错误修改,这一属性被设置为privae的,不过可以通过size()获取。

   下面,对ArrayList的初始以及其列表对象的增加和删除等三种情况下的size自动改变机制进行分析。

   1、初始Capacity和size值。

   从上面给出的ArrayList构造方法源码中,我们不难看出Capacity初始值(initialCapacity)可以由用户直接指定或由用户指定的Collection集合存

储的对象数目确定,如果没有指定,系统默认为10。而size的被声明为int型变量,默认为0,当用户指定Collection创建ArrayList时,size值等于

initialCapacity。


   2、add()方法

    该方法的源码如下:

    public boolean add(E e) {
        ensureCapacityInternal(size + 1);
        elementData[size++] = e;//添加对象时,自增size
        return true;
    }

    方法中调用的ensureCapacityInternal主要用来调整容量,修改elementData数组的指向。其中涉及到3个方法的调用,其核心在于grow方法:
  
    private void ensureCapacityInternal(int minCapacity) {
        modCount++;//定义于ArrayList的父类AbstractList,用于存储结构修改次数
        // overflow-conscious code
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    } 

    private void grow(int minCapacity) {
        // overflow-conscious code
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);//新容量扩大到原容量的1.5倍,右移一位相关于原数值除以2。
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        // minCapacity is usually close to size, so this is a win:
        elementData = Arrays.copyOf(elementData, newCapacity);
    }

    private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;//MAX_ARRAY_SIZE和Integer.MAX_VALUE为常量,详细请参阅下面的注解
    }
 
   通过以上代码,我们可知java自动增加ArrayList大小的思路是:向ArrayList添加对象时,原对象数目加1如果大于原底层数组长度,则以适当长度新

建一个原数组的拷贝,并修改原数组,指向这个新建数组。原数组自动抛弃(java垃圾回收机制会自动回收)。size则在向数组添加对象,自增1。


   注解:

    //定义于该类的常量,用来分配数组的size最大值。一些 VMs在数组里保留字头,试图分配更大数组时可能导致OutOfMemoryError:被请求数组的

size超出VM界限。
    private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

   //在java.lang.Integer类中常量MIN_VALUE、MAX_VALUE如下:
   public static final int   MIN_VALUE = 0x80000000;//整型取值区间下界:-2147483648
   public static final int   MAX_VALUE = 0x7fffffff;//整型取值区间上界:2147483647

  //在java.util.AbstractList中modCount定义如下:
  protected transient int modCount = 0;

    3、remove()方法

    该重构方法其一源码如下(其它的就不累述了):
    public E remove(int index) {
        rangeCheck(index);

        modCount++;
        E oldValue = elementData(index);

        int numMoved = size - index - 1;
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);//将后面的列表对象前移
        elementData[--size] = null; // 数组前移一位,size自减,空出来的位置置null,具体的对象的销毁由Junk收集器负责
        return oldValue;
    }

    private void rangeCheck(int index) {//边界检查
        if (index < 0 || index >= this.size)
           throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
    }

    E elementData(int index) {//获取指定index所在位置的对象
        return (E) elementData[index];
    }


    通过remove()源码的学习,我们不难看出,其改变ArrayList大小的核心与add()方法相似,都是同数组拷贝。

    另外,如果确有必要,用户也可以指定ArrayList实例的容量,可以有效的降低时间成本。它是通过调用ensureCapacityInternal来实现的,源代码

如下:
    public void ensureCapacity(int minCapacity) {
        if (minCapacity > 0)
            ensureCapacityInternal(minCapacity);
    }

    因为size为private的,java给出方法来访问它:
    public int size() {
         checkForComodification();
         return this.size;
    }


    综上所述,在用户向ArrayList追加对象时,Java总是要先计算容量(Capacity)是否适当,若容量不足则把原数组拷贝到以指定容量为长度创建的

新数组内,并对原数组变量重新赋值,指向新数组。在这同时,size进行自增1。在删除对象时,先使用拷贝方法把指定index后面的对象前移1位(如果

有的话),然后把空出来的位置置null,交给Junk收集器销毁,size自减1,即完成了。

分享到:
评论

相关推荐

    用C语言模拟ArrayList

    在C语言中,ArrayList是一种常见的数据结构,它模拟了Java或.NET等高级语言中的动态数组。ArrayList提供了在数组中添加、删除和查找元素的便利操作,而无需预先知道数组的大小。下面,我们将深入探讨如何用C语言实现...

    模拟arraylist底层实现

    模拟ArrayList的size方法 size方法是用于获取集合中的元素个数的方法。在MyArrayList类中,我们可以看到size方法的实现逻辑。size方法的主要作用是返回集合中的元素个数。因此,在size方法中,我们直接返回size变量...

    ArrayList源码Jdk1.8

    与固定大小的数组不同,`ArrayList`可以随着元素的增加而自动扩展其容量。在JDK 1.8中,`ArrayList`实现了`List`接口,并且允许包含任何类型的对象,包括`null`。 #### 核心特性 - **高效性**: `ArrayList`中的大...

    jni操作arraylist对象

    在这个主题中,我们将深入探讨如何在JNI中操作ArrayList对象并添加一个int类型的数据。 首先,我们需要理解ArrayList在Java中的本质。ArrayList是Java集合框架中的一个重要类,它实现了List接口,用于存储可变大小...

    C# Array和ArrayList,List区别

    `ArrayList` 类位于 `System.Collections` 命名空间中,它可以存储不同类型的数据,并且大小可以动态改变。创建一个 `ArrayList` 的基本语法如下: ```csharp ArrayList al = new ArrayList(); ``` **添加与访问**...

    深入Java集合学习系列(三):ArrayList实现原理

    如果多个线程同时访问同一个ArrayList实例,并且其中至少有一个线程在结构上修改了列表(指的是改变列表的大小),那么必须在外部保持同步,以避免数据不一致的问题。这种情况下,可以通过诸如Collections....

    arraylist用法

    如果`Count`超过了`Capacity`,`ArrayList`会自动增加其容量以适应更多的元素。 4. **添加、删除和插入操作** - `Add`和`AddRange`:分别向`ArrayList`的末尾添加一个元素或一系列元素。 - `Remove`、`RemoveAt`...

    你必须知道的C# ArrayList

    - ArrayList内部维护了一个Object类型的数组,当添加或删除元素时,它会自动调整数组大小以适应变化。 2. **ArrayList的主要方法** - `Add(object item)`: 向ArrayList末尾添加一个元素。 - `Insert(int index, ...

    ArrayList类操作程序实例

    1. `size()`:返回ArrayList中的元素数量。 2. `ensureCapacity(int minCapacity)`:确保ArrayList的容量至少为minCapacity,如果不足则扩容。 3. `trimToSize()`:将ArrayList的容量调整为其实际元素的数量。 八、...

    数组模仿ArrayList

    ArrayList则是通过动态数组来模拟可变大小的列表,当添加或删除元素时,它会自动调整数组的大小以适应需求。因此,模仿ArrayList的第一步就是创建一个可扩展的数组。 ```java public class MyArrayList&lt;T&gt; { ...

    模拟java ArrayList Iterator

    在Java编程语言中,ArrayList是集合框架中的一种重要数据结构,它是一个动态数组,可以存储任意类型的对象。ArrayList提供了一种高效的方式来管理大量的元素,并且提供了迭代器(Iterator)来遍历这些元素,使得我们...

    c版的arraylist

    在Java编程语言中,`ArrayList`是`java.util`包中的一个重要集合类,它提供了动态数组的功能。这个数据结构允许我们存储、访问和管理一组元素。而在C语言中,由于没有内置的类似集合的数据类型,程序员需要自定义...

    C++ArrayList

    此外,`std::vector`是C++标准库中一个功能强大的动态数组,它自动处理内存管理,但在仿照Java的ArrayList时,我们可能需要手动实现这些功能,以便更好地理解数据结构的底层运作。 1. **类定义**:ArrayList类应该...

    C#数组与Arraylist

    ArrayList可以存储任何类型的对象,无需预先指定长度,因为它会自动调整大小以适应添加或删除元素的需求: ```csharp ArrayList list = new ArrayList(); list.Add("Hello"); list.Add(123); // ... ``` ArrayList...

    浅析ArrayList内部实现

    浅析ArrayList内部实现 ArrayList是Java集合框架中的一种常用数据结构,能够存储任意多个对象,并且可以自由扩展,弥补了数组的定长的缺陷。下面我们将深入探讨ArrayList的内部实现机理。 ArrayList的内部实现机理...

    C#ArrayList的详细用法

    1. 支持自动改变大小的功能:ArrayList 可以根据需要自动调整数组的大小,避免了数组固定大小的限制。 2. 可以灵活的插入元素:ArrayList 提供了多种插入方法,例如 Add、Insert、InsertRange 等,使得开发者可以...

    对Java ArrayList的自动扩容机制示例讲解

    对Java ArrayList的自动扩容机制示例讲解 在Java中,ArrayList是最常用的集合类之一,它提供了动态扩容的功能,能够自动地根据添加元素的数量来调整内部数组的大小,以满足元素的存储需求。今天,我们将深入探讨...

    如何遍历ArrayList

    i &lt; arrayList.size(); i++){ arrayList.get(i); // ......... } ``` 在上面的代码中,我们使用for循环来遍历ArrayList中的元素。在每次循环中,我们使用arrayList.get(i)方法来获取第i个元素。 使用foreach...

    arrayList排序

    - `Collections.sort()`方法会改变原ArrayList的顺序,如果不想改变原集合,可以先复制一份ArrayList再进行排序。 - 对于大量数据的排序,ArrayList的性能可能不如使用LinkedList,因为ArrayList排序时涉及到大量...

    C语言版的ArrayList

    C语言版的ArrayList,具有ArrayList的基本方法增加、插入、删除、自动扩容等。

Global site tag (gtag.js) - Google Analytics