阅读 121

ArrayList——源码分析

ArrayList源码简单分析

public class ArrayList<E> extends AbstractList<E>
        implements List<E>, RandomAccess, Cloneable, java.io.Serializable复制代码

ArrayList实现了RandomAccess,Cloneable,Serializable说明ArrayList是支持随机访问,可克隆,支持序列化的。其继承AbstractList,实现了List,说明其支持列表的相关操作,如添加、删除、判断是否为空、排序等等。

常用方法分析

构造方法

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);
    }
}复制代码
public ArrayList() {
    this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}复制代码

有参构造方法,根据传入的参数设置ArrayList底层数组的长度,如果传参为0的话数组为空数组。无参即默认使用空数组。

get()

public E get(int index) {
    rangeCheck(index);

    return elementData(index);
}复制代码

判断是否下标越界,没有的话就根据数组支持随机访问的特性获得该下标下的值。

set()

public E set(int index, E element) {
    rangeCheck(index);

    E oldValue = elementData(index);
    elementData[index] = element;
    return oldValue;
}复制代码

同上,检查下标,利用随机访问的特性去更新某一下标的值,将旧值返回。

add()

public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    elementData[size++] = e;
    return true;
}复制代码
private void ensureCapacityInternal(int minCapacity) {
    ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}复制代码
// 当数组为空数组时,返回默认数组长度为10,否则返回minCapacity
private static int calculateCapacity(Object[] elementData, int minCapacity) {
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        return Math.max(DEFAULT_CAPACITY, minCapacity);
    }
    return minCapacity;
}复制代码
// 判断是否需要扩容
private void ensureExplicitCapacity(int minCapacity) {
    modCount++;

    // overflow-conscious code
    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}复制代码

grow()方法主要是对数组的扩容操作,扩容后的数组大小是原数组的1.5倍,同时将旧数组的内容拷贝到新数组上。

综上,每次添加会对size值加一,然后判断是否需要扩容,最后将数据添加到数组尾部。

public void add(int index, E element) {
    rangeCheckForAdd(index);

    ensureCapacityInternal(size + 1);  // Increments modCount!!
    System.arraycopy(elementData, index, elementData, index + 1,
                     size - index);
    elementData[index] = element;
    size++;
}复制代码

根据add(E e) 分析可知,add(int index, E element)的逻辑为判断index的下标是否越界,然后判断是否需要进行扩容操作,接着根据数组的特性,将index + 1 ~ size的数组数据进行向后迁移。最后将数据插入到指定位置。

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; // clear to let GC do its work

    return oldValue;
}复制代码

同add(int index, E element)类似,判断越界,迁移,最后将数组末尾的数据设为null,交由GC去执行回收。

public boolean remove(Object o) {
    if (o == null) {
        for (int index = 0; index < size; index++)
            if (elementData[index] == null) {
                fastRemove(index);
                return true;
            }
    } else {
        for (int index = 0; index < size; index++)
            if (o.equals(elementData[index])) {
                fastRemove(index);
                return true;
            }
    }
    return false;
}复制代码

遍历数组,进行相等判断,然后进行数据的迁移。

内部类Itr

ArrayList内嵌内部类Itr,支持迭代器。(这部分内容,读者自行了解,不加赘述)

private class Itr implements Iterator<E>复制代码

内部类SubList

private class SubList extends AbstractList<E> implements RandomAccess


作者:不二鑫
链接:https://juejin.cn/post/7019565835538677768


文章分类
后端
版权声明:本站是系统测试站点,无实际运营。本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 XXXXXXo@163.com 举报,一经查实,本站将立刻删除。
相关推荐