Java ArrayList 工作原理及实现

本人花费半年的时间总结的《Java面试指南》已拿腾讯等大厂offer,已开源在github ,欢迎star!

本文GitHub https://github.com/OUYANGSIHAI/JavaInterview 已收录,这是我花了6个月总结的一线大厂Java面试总结,本人已拿大厂offer,欢迎star

原文链接:blog.ouyangsihai.cn >> Java ArrayList 工作原理及实现

点击上方“Java知音”,选择“置顶公众号”

技术文章第一时间送达!

作者:兵小志大
链接:www.cnblogs.com/try-better-tomorrow

链接:www.cnblogs.com/try-better-tomorrow

公众号注:集合源码解析,点击文末阅读原文直达


1.概述

关于Java集合小抄中的描述:

http://calvin1978.blogcn.com/articles/collection.html

以数组实现。节约空间,但数组有容量限制。超出限制时会增加50%容量,用System.arraycopy()复制到新的数组,因此最好能给出数组大小的预估值。默认第一次插入元素时创建大小为10的数组。

按数组下标访问元素—get(i)/set(i,e) 的性能很高,这是数组的基本优势。

直接在数组末尾加入元素—add(e)的性能也高,但如果按下标插入、删除元素—add(i,e), remove(i), remove(e),则要用System.arraycopy()来移动部分受影响的元素,性能就变差了,这是基本劣势。

然后再来学习一下官方文档:

Resizable-array implementation of the List interface. Implements all optional list operations, and permits all elements, including null. In addition to implementing the List interface, this class provides methods to manipulate the size of the array that is used internally to store the list. (This class is roughly equivalent to Vector, except that it is unsynchronized.)

ArrayList是一个相对来说比较简单的数据结构,最重要的一点就是它的自动扩容,可以认为就是我们常说的“动态数组”。

来看一段简单的代码:


ArrayListString list = new ArrayListString();

list.add("语文: 99");

list.add("数学: 98");

list.add("英语: 100");

list.remove(0);

在执行这四条语句时,是这么变化的:

其中,add操作可以理解为直接将数组的内容置位,remove操作可以理解为删除index为0的节点,并将后面元素移到0处。

2. add函数

当我们在ArrayList中增加元素的时候,会使用add函数。他会将元素放到末尾。具体实现如下:


public boolean add(E e) {

    ensureCapacityInternal(size + 1);  // Increments modCount!!

    elementData[size++] = e;

    return true;

}

我们可以看到他的实现其实最核心的内容就是ensureCapacityInternal。这个函数其实就是自动扩容机制的核心。我们依次来看一下他的具体实现


private void ensureCapacityInternal(int minCapacity) {
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
    }
    ensureExplicitCapacity(minCapacity);
}

private void ensureExplicitCapacity(int minCapacity) {
    modCount++;

    // overflow-conscious code
    if (minCapacity - elementData.length  0)
        grow(minCapacity);
}

private void grow(int minCapacity) {
    // overflow-conscious code

    int oldCapacity = elementData.length;
    // 扩展为原来的1.5倍

    int newCapacity = oldCapacity + (oldCapacity  1);
    // 如果扩为1.5倍还不满足需求,直接扩为需求值

    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);
}

也就是说,当增加数据的时候,如果ArrayList的大小已经不满足需求时,那么就将数组变为原长度的1.5倍,之后的操作就是把老的数组拷到新的数组里面。

例如,默认的数组大小是10,也就是说当我们add10个元素之后,再进行一次add时,就会发生自动扩容,数组长度由10变为了15具体情况如下所示:

3.set和get函数

Array的put和get函数就比较简单了,先做index检查,然后执行赋值或访问操作:


public E set(int index, E element) {
    rangeCheck(index);
    E oldValue = elementData(index);
    elementData[index] = element;
    return oldValue;
}

public E get(int index) {
    rangeCheck(index);
    return elementData(index);
}

4.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);
    // 把最后的置null
    elementData[--size] = null; // clear to let GC do its work
    return oldValue;

}

注释很清楚:

Removes the element at the specified position in this list. Shifts any subsequent elements to the left (subtracts one from their indices).

参考资料

  • Class ArrayList

    http://docs.oracle.com/javase/8/docs/api/java/util/ArrayList.html

  • ArrayList其实就那么一回事儿之源码浅析

    http://www.cnblogs.com/dongying/p/4013271.html

  • 关于ArrayList http://my.oschina.net/tunie/blog/122530
  • http://docs.oracle.com/javase/8/docs/api/java/util/ArrayList.html

    http://www.cnblogs.com/dongying/p/4013271.html

    http://my.oschina.net/tunie/blog/122530

    看完本文有收获?请转发分享给更多人

    Java ArrayList 工作原理及实现
    本人花费半年的时间总结的《Java面试指南》已拿腾讯等大厂offer,已开源在github ,欢迎star!

    本文GitHub https://github.com/OUYANGSIHAI/JavaInterview 已收录,这是我花了6个月总结的一线大厂Java面试总结,本人已拿大厂offer,欢迎star

    原文链接:blog.ouyangsihai.cn >> Java ArrayList 工作原理及实现


     上一篇
    HashMap的put和get方法原理 HashMap的put和get方法原理
    来源:Sam哥哥https://blog.csdn.net/linsongbin1公众号注:文末阅读原文可直达java开发学习网站 公众号注:文末阅读原文可直达java开发学习网站 概述 JAVA中的数组,在添加或者删除元素的时候,都会
    下一篇 
    Java之Map遍历方式性能分析——ketSet 与 entrySet Java之Map遍历方式性能分析——ketSet 与 entrySet
    来源:徐亦快https://blog.csdn.net/weixin_42280517公众号注:文末阅读原文可直达java开发学习网站 https://blog.csdn.net/weixin_42280517 keySet():将Ma