arraylist原理解析

mvilplss mvilplss     2022-09-06     578

关键词:

简介

ArrayList就是动态数组,用MSDN中的说法,就是Array的复杂版本,它提供了动态的增加和减少元素,实现了ICollection和IList接口,灵活的设置数组的大小等好处

有图有码

图为手工画的,有点丑见谅 _!

  1. 初始化集合ArrayList list = new ArrayList();
    因为使用无参构造时候集合容器为空,所以无任何空位。
  2. 第一次添加元素 add("a") 第一次添加元素时候,检测容器为空,根据默认容量10进行初始化容器。然后将元素放置到第一个空位中。 初始化容器:image
    增加一个元素:image
  3. 第十一次添加元素 add("k") 第十一次添加元素,发现元素超出容量,所以进行一次扩容,扩容后的大小为原容量加原容量的二分之一,即为15;然后将元素放置到第是一个空位中。
    增加容量:image
    增加一个元素:image
  4. 移除期中一个元素 remove(3) 移除第三个元素,将数组从第四个元素到最后一个元素放置到第三个元素开始到最后,然后将数组的最后一位元素设为null。 image

源码分析

构造方法

//无参构造方法,初始化elementData为{}
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
//元素数组
transient Object[] elementData;
//集合大小
private int size;
//默认容量
private static final int DEFAULT_CAPACITY = 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);
        }
    }
//带有初始化数据的构造
 public ArrayList(Collection<? extends E> c) {
        elementData = c.toArray();
        if ((size = elementData.length) != 0) {
            // c.toArray might (incorrectly) not return Object[] (see 6260652)
            // 因为c.toArray返回的不一定为Object[]
            // 如Object[] objs = new String[]{""};
            // objs[0]=new Object();//java.lang.ArrayStoreException: java.lang.Object
            //经测试 在1.8被修复了^_^
            if (elementData.getClass() != Object[].class)
                elementData = Arrays.copyOf(elementData, size, Object[].class);
        } else {
            // replace with empty array.
            this.elementData = EMPTY_ELEMENTDATA;
        }
    }

方法及源码

  • add
//增加一个元素
public boolean add(E e) {
        //确保有空余的容量,否则则增加容量
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        elementData[size++] = e;
        return true;
    }
//指定位置插入一个元素
public void add(int index, E element) {
        //检测下标是否越界,否则 throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
        rangeCheckForAdd(index);
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        //拷贝数组,为新元素腾出一个空位
        System.arraycopy(elementData, index, elementData, index + 1,
                         size - index);
        elementData[index] = element;
        size++;
    }
//确保有空余的容量,否则则增加容量
private void ensureCapacityInternal(int minCapacity) {
//如果集合为空,则取默认和当前size+1较大的值
        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;
        //新容量=原来大小+原来大小/2,也就是说扩容原来大小的一半。
        //备注: x>>1=x/2
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        //如果新容量还不够用,设容量为所需容量
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
        //如过所需容量大于最大容量Integer.MAX_VALUE - 8,则设置超大容量
        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);
    }
//设置超大容量,capacity=Integer.MAX_VALUE=2147483647    
private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }
  • remove
//首先根据对象循环对比,找出第一个相等的对象的下标
//然后通过fastRemove方法进行快速移除
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;
    }
//根据下标进行移除,返回被移除的元素
public E remove(int index) {
        //检查下标是否越界
        rangeCheck(index);

        modCount++;
        E oldValue = elementData(index);
        //计算出要拷贝的对象个数
        int numMoved = size - index - 1;
         //移除的原理就是将指定下标的后面元素全部向前移动一步,将末端元素设为null
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // clear to let GC do its work

        return oldValue;
    }
//快速移除方法
private void fastRemove(int index) {
        modCount++;
        int numMoved = size - index - 1;
        //移除的原理就是将指定下标的后面元素全部向前移动一步,将末端元素设为null
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // clear to let GC do its work
    }
  • 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;
    }
  • clear
//清楚集合,循环元素,将每个元素设置为null
public void clear() {
        modCount++;

        // clear to let GC do its work
        for (int i = 0; i < size; i++)
            elementData[i] = null;

        size = 0;
    }

总结

通过源码分析,ArrayList集合就是通过System.arraycopy方法将普通数组Object[]包装为一个动态数组,实现数组的增删改查。

  • 优点
    1、修改元素和通过下标查询元素效率高
    2、集合是有顺序的
  • 缺点
    1、删除元素效率低,因为要通过拷贝数组来实现
    2、大量新增效率低,因为大量新增的时候要不断进行扩容和数组的拷贝
    3、清除集合效率低,因为清除功能是通过循环数组进行清除的
    4、移除元素后,容量有大量剩余,需要手动调用trimToSize进行清理

其他

1、使用时候如果知道预期容量,建议设定容量,避免不断扩容影响效率。
2、建议改进清除操作,避免使用循环进行清除。

参考

更多文章请关注微信 !!!!!!

高并发容器copyonwritearraylist原理解析(代码片段)

CopyOnWriteArrayList实现arraylist多线程数据安全的方式jdk提供的Collections.SynchronizedList()所有方法进行添加synchronized块publicvoidadd(intindex,Eelement)synchronized(mutex)list.add(index,element);使用reentrantLock自己对add时& 查看详情

arraylist源码解析

ArrayList源码解析ArrayList简介:ArrayList是list接口的一个常用实现类。它的对象可以认为是一维数组的“类版本”。我们很快就可以看到,ArrayList对象可以看做是一维数组的改良版本。类似于数组,ArrayList对象支持元素的随机访问... 查看详情

arraylist实现原理分析(代码片段)

ArrayList使用的存储的数据结构ArrayList的初始化ArrayList是如何动态增长ArrayList如何实现元素的移除ArrayList小结ArrayList是我们经常使用的一个数据结构,我们通常把其用作一个可变长度的动态数组使用,大部分时候,可以替代数组的... 查看详情

arraylist解析

我们了解一个接口,最好的切入点是API官方文档,所以首先看下ArrayList接口的接口API:publicclassArrayList<E>extendsAbstractList<E>implmentsList,RandomAccess,Cloneable,Serializable我们首先看到ArrayList接口继承自抽象类AbstractList;关于Abstract 查看详情

arraylist内部实现原理

ArrayList内部实现原理javalist首先,我们new一个对象list集合 List<String>list=newArrayList<>();我们知道对象的创建离不开构造方法,因此我们查看ArrayList源码的时候先看其构造方法 privatestaticfinalObject[]DEFAULTCAPACITY_EMPTY_ELEMENTDATA={}... 查看详情

arraylist解析(代码片段)

ArrayList属性//默认长度privatestaticfinalintDEFAULT_CAPACITY=10;//底层是以数组格式存储privatestaticfinalObject[]EMPTY_ELEMENTDATA=;privatestaticfinalObject[]DEFAULTCAPACITY_EMPTY_ELEMENTDATA=;transientObject[]ele 查看详情

arraylist源码解析

 这篇文章主要看ArrayList的Iterator和ListIterator的实现。1.Iterator和类Itr当我们调用iterator方法时返回一个Iterator。/***Returnsaniteratorovertheelementsinthislistinpropersequence.**<p>Thereturnediteratoris<ahref="#f 查看详情

arraylist实现原理

继承关系:Object-Collection-AbstractList-ArrayListpublicclassArrayList<E>extendsAbstractList<E>implementsList<E>,RandomAccess,Cloneable,java.io.Serializable{}size()代表ArrayList实际存储元素的数量ele 查看详情

arraylist源码解析

add操作:privatetransientObject[]elementData;privatestaticfinalintDEFAULT_CAPACITY=10;publicArrayList(){super();this.elementData=EMPTY_ELEMENTDATA;}publicArrayList(intinitialCapacity){super();if(initialC 查看详情

list源码解析之arraylist源码分析

ArrayList简介ArrayList是基于数组实现的,是一个动态扩展的数组,容量可自动增长。ArrayList是非线程安全的,只能在单线程环境下使用,多线程环境考虑使用Collections.synchronizedList(Listlist)函数返回一个线程安全的ArrayList类,也可以... 查看详情

浅谈arraylist的底层扩容的原理(代码片段)

ArrayList扩容机制的源码详解一:ArrayList的构造函数:ArrayList的构造函数源码有三种:先来看看ArrayList底层定义的一些变量的含义:/**Defaultinitialcapacity*默认的容量大小*/privatestaticfinalintDEFAULT_CAPACITY=10;/**Sharedemptyarrayinstanceusedforempty... 查看详情

arraylist源码解析

...误或描述不准确的地方,还请大家指出。回到顶部1.位置ArrayList位于java.util包中。1packagejava.util;23importjava.util.function.Consumer;4importjava.util.f 查看详情

java集合干货系列-arraylist源码解析

前言今天来介绍下ArrayList,在集合框架整体框架一章中,我们介绍了List接口,ArrayList继承了AbstractList,实现了List。ArrayList在工作中经常用到,所以要弄懂这个类是极其重要的。构造图如下:蓝色线条:继承绿色线条:接口实现正... 查看详情

arraylist源码解析

 1.isEmpty()   如果此列表中没有元素,则返回true/***Returns<tt>true</tt>ifthislistcontainsnoelements.*/publicbooleanisEmpty(){returnsize==0;}  判断ArrayList是否为空,size为0时,即不包含任何成员时为空,返回true。2.index 查看详情

arraylist源码解析

巩固下基础,阅读下jdk的源码,这篇文章是来介绍下ArrayList的实现。1.ArrayList概述    List 接口的大小可变数组的实现,位于API文档的java.util.ArrayList<E>。实现了所有可选列表操作,并允许包括null在内的所有元素。除... 查看详情

arraylist实现原理(代码片段)

...链接:http://www.cnblogs.com/ITtangtang/p/3948555.html#sum一、 ArrayList概述:    ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长,类似于C语言中的动态申请内存,动态增长内存。  ArrayList不是线程... 查看详情

java集合之list---arraylist解析

本章节主要讲述ArrayList集合知识,主要包括ArrayList结构类型,集合特点,源码解析等。欢迎持续关注阅读,一起学习,共同交流(477819525君羊)。1ArrayList类层次结构:650)this.width=650;"src="http://img.blog.csdn.net/20170224154152480?watermark/2/text... 查看详情

将ArrayList解析为日期[重复]

】将ArrayList解析为日期[重复]【英文标题】:parseArrayListtoDate[duplicate]【发布时间】:2015-12-2811:46:12【问题描述】:我有一个带有字符串的数组列表,我想将它转换为日期。似乎“解析”不能用于执行此操作,在以下代码中:List&l... 查看详情