Java 集合框架 ArrayList

本文深入解析ArrayList的内部工作机制,包括其基于数组的实现方式、线程安全性、动态扩容机制以及添加和删除元素的具体步骤。分析了ArrayList在不同操作下的性能特点,指出其适用于尾部操作和随机访问的场景。
摘要由CSDN通过智能技术生成

一、 ArrayList 概述:

 ArrayList 源码

public class ArrayList<E> extends AbstractList<E>
       implements List<E>, RandomAccess, Cloneable, java.io.Serializable {}
  •  ArrayList 是基于数组实现的,是一个动态数组,其容量能自动增长。
  •  ArrayList 不是线程安全的,只能用在单线程环境下,多线程环境下可以考虑用 Collections.synchronizedList(List) 函数返回一个线程安全的 ArrayList 类,也可以使用concurrent 并发包下的 CopyOnWriteArrayList 类。
  • ArrayList 实现了 Serializable 接口,因此它支持序列化,能够通过序列化传输;实现了RandomAccess 接口,支持快速随机访问,实际上就是通过下标序号进行快速访问;实现了 Cloneable 接口,能被克隆。
  • 每个 ArrayList 实例都有一个容量,该容量是指用来存储列表元素的数组的大小。它总是至少等于列表的大小。随着向 ArrayList 中不断添加元素,其容量也自动增长。自动增长会带来数据向新数组的重新拷贝,因此,如果可预知数据量的多少,可在构造 ArrayList时指定其容量。在添加大量元素前,应用程序也可以使用 ensureCapacity 操作来增加   ArrayList 实例的容量,这可以减少递增式再分配的数量。
  • 此类的 iterator 和 listIterator 方法返回的 iterator 是快速失败(fail-fast)的:如果在创建迭代器后修改了列表的结构,除了通过迭代器自己的 remove 或 add 方法,迭代器将抛出 ConcurrentModificationException 。迭代器的快速失败行为应该仅用于检测错误。

二、ArrayList 的主要成员变量:

    /**
     * Default initial capacity.
     */
    private static final int DEFAULT_CAPACITY = 10;

    /**
     * Shared empty array instance used for empty instances.
     */
    private static final Object[] EMPTY_ELEMENTDATA = {};

    /**
     * Shared empty array instance used for default sized empty instances. We
     * distinguish this from EMPTY_ELEMENTDATA to know how much to inflate when
     * first element is added.
     */
    private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};

    /**
     * The array buffer into which the elements of the ArrayList are stored.
     * The capacity of the ArrayList is the length of this array buffer. Any
     * empty ArrayList with elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA
     * will be expanded to DEFAULT_CAPACITY when the first element is added.
     */
    transient Object[] elementData; // non-private to simplify nested class access

    /**
     * The size of the ArrayList (the number of elements it contains).
     */
    private int size;
     
    /**
     * 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;

三、ArrayList 的构造方法

    // 构造一个初始容量为 10 的空列表
    public ArrayList() {
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
    }

    // 构造一个具有指定初始容量的空列表
    public ArrayList(int initialCapacity) {
        if (initialCapacity > 0) {
            this.elementData = new Object[initialCapacity];
        } else if (initialCapacity == 0) {
            this.elementData = EMPTY_ELEMENTDATA;
        } else {
            throw new IllegalArgumentException("Illegal Capacity: "
                                               + initialCapacity);
        }
    }

四、ArrayList add() 方法

    /**
     * 在add()方法中主要完成了三件事:
     * (1)确保能够将将要添加到集合中的元素添加到集合中,
     *      即确保ArrayList的容量(判断是否需要扩容);
     * (2)将元素添加到elementData数组的指定位置;
     * (3)将集合中实际的元素个数加 1。
     */
    // 1.1 将指定的元素添加到此列表的末尾。
    public boolean add(E e) {
        ensureCapacityInternal(size + 1); // Increments modCount!!
        elementData[size++] = e;
        return true;
    }

    // 1.2 先初始化ArrayList的容量为10,
    private void ensureCapacityInternal(int minCapacity) {
        // 仅第一次添加元素时执行,并将数组容量初始化为10
        // 判断原数组是否为空数组
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }
        ensureExplicitCapacity(minCapacity); // 判断是数组否需要扩容
    }

	// 1.3 判断是数组否需要扩容
    private void ensureExplicitCapacity(int minCapacity) {
        modCount++; // 记录数组结构修改的次数
        // 第一次添加元素、后续数组需要扩容时执行
        // 当前数组需要的最小容量与原来数组容量比较看是否需要扩容
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }

	// 1.4 数组扩容
    private void grow(int minCapacity) {
        int oldCapacity = elementData.length; // 原来数组容量
        // 扩容机制:1.5倍扩容
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        if (newCapacity - minCapacity < 0) // 仅第一次添加元素时执行
            newCapacity = minCapacity;
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        // 扩容后的新数组
        elementData = Arrays.copyOf(elementData, newCapacity);
    }
	
	// 1.5 数组的最大容量
    private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }

小结:

ArrayList 的扩容主要发生在向 ArrayList 集合中添加元素的时候。
由 add() 方法的分析可知:添加前必须确保集合的容量充足,能够放下添加的元素。

主要经历了以下几个阶段:
① 在 add() 方法中调用 ensureCapacityInternal(size+1) 方法来确定:集合成功添加元素的最小集合容量 minCapacity 的值。参数为size+1,代表的含义是如果集合添加元素成功后,集合中的实际元素个数。换句话说,集合为了确保添加元素成功,那么集合的最小容量minCapacity 应该是 size+1。在 ensureCapacityInternal 方法中,首先判断 elementData 是否为默认的空数组,如果是,minCapacity为minCapacity 与集合默认容量大小中的较大值。
② 调用 ensureExplicitCapacity(minCapacity) 方法来确定:集合为了确保添加元素成功是否需要对现有的元素数组进行扩容。首先将结构性修改计数器加1;然后判断 minCapacity 与当前数组元素的长度大小,如果 minCapacity 比当前数组元素的长度大小大的时候需要扩容,进入第三阶段。
③ 如果需要对现有的元素数组进行扩容,则调用 grow(minCapacity) 方法,参数 minCapacity 表示集合为了确保成功添加元素的最小容量。在扩容的时候,首先将原元素数组的长度增大 1.5 倍(oldCapacity(oldCapacity >> 1)),然后对扩容后的容量与 minCapacity 进行比较:
    ① 新容量小于 minCapacity,则将新容量设为 minCapacity;
    ② 新容量大于 minCapacity,则指定新容量。
最后将旧数组拷贝到扩容后的新数组中。

五、ArrayList 的 remove(int index) 方法

	// 移除此列表中指定位置的元素。将任何后续元素向左移动(从它们的索引中减去一个)。
	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);
        // 赋值为空,有利于进行GC
        elementData[--size] = null; // clear to let GC do its work
        return oldValue; // 返回移除的旧值
    }

小结

1、根据索引删除
   由于底层是数组存储,为了保障数组的连续性,删除数据会导致数据移动,删除位置越靠前,移动数据越多。

2、根据元素对象删除
   实际是遍历整个 ArrayList,使用对象的 equals() 判断是否相等,先查出该对象所在的索引位置,然后根据索引位置删除元素。

六、ArrayList的优缺点

ArrayList的优点

  • ArrayList 底层以数组实现,是一种随机访问模式,再加上它实现了RandomAccess 接口,因此查找的时候非常快。
  • ArrayList 在顺序添加一个元素的时候非常方便,只是往数组里面添加了一个元素而已。根据下标遍历元素,效率高。
  • 可以自动扩容,默认为每次扩容为原来的1.5倍

ArrayList的缺点

  • 插入和删除元素的效率不高。根据元素下标查找元素需要遍历整个元素数组,效率不高。
  •  线程不安全。

注意:ArrayList 比较适合尾部添加、删除、随机访问的场景。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值