ArrayList是java中的线性结构的一种表示方法,在java中使用频率非常高,下面来一步一步分析其底层的实现。(JDK1.8)

一、构造函数

ArrayList的构造函数有三个,分别如下,

我们最常使用的的无参的构造函数,那么无参的构造函数是如何定义的那,

public ArrayList() {
this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

从上面看到无参构造函数,实际上是把elementData赋予了默认容量的一个数组DEFAULTCAPACITY_EMPTY_ELEMENTDATA;

 private static final Object[] 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);
}
}

从上面可以看出首先判断给定的初始容量是否大于0,如果大于0则使用此值生成Object数组。如果等于0则和无参构造函数是一样的。如果小于0则抛出异常。在实际开发中强烈建议使用带参数的构造函数(为什么下面说)。
最后一个构造函数如下,

public ArrayList(Collection<? extends E> c) {
elementData = c.toArray();
if ((size = elementData.length) != 0) {
// c.toArray might (incorrectly) not return Object[] (see 6260652)
if (elementData.getClass() != Object[].class)
elementData = Arrays.copyOf(elementData, size, Object[].class);
} else {
// replace with empty array.
this.elementData = EMPTY_ELEMENTDATA;
}
}

从上面的代码中可以看出,把集合类c调用toArray方法,返回其数组赋给elementData,然后对elementData进行判断如果长度不为0,这判断elementData的类型是否为Object,如果不是则使用Object进行替换,统一处理成Obejct的类型,因为在java中所有的类的父类都是Object。
二、成员变量

在ArrayList中主要的成员变量有elementData和size,elementData代表底层ArrayList的存储结构,即ArrayList是用数组来保存数据的;size指的是ArrayList中元素的个数(和elementData的长度区分开)。其他的成员变量像默认的初始容量为10。ArrayList在使用无参构造函数的时候分配内存的时候采用了懒加载的模式,即使用new ArrayList的时候底层不会分配空间,在真正向ArrayList中放元素的时候才会真正初始化数组长度。

三、常用方法

1、set/add

set和add方法的不同之处在于set方法会返回被覆盖的值,而add则不会

set(int index,E element)

在index处插入元素element,其具体实现如下

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

从上面的代码中可以看出,首先需要判断index是否合法,

 private void rangeCheck(int index) {
if (index >= size)
throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
}

也就是说如果index大于ArrayList中元素的个数,则抛出异常,这里有个疑问ArrayList是使用数组存储,默认长度是10,那么向ArrayList中添加一个元素后其数组的长度为10,ArrayList中的元素为1,为什么不允许index大于size插入那。
在判断完index不大于size后,把要插入的元素放在数组中下标为index的位置,返回之前的元素。

add(E e)

向ArrayList中插入元素e返回boolean类型,表示是否插入成功。

public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
}

可以看到首先要确保数组的容量,

private void ensureCapacityInternal(int minCapacity) {
ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
private static int calculateCapacity(Object[] elementData, int minCapacity) {
if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
return Math.max(DEFAULT_CAPACITY, minCapacity);
}
return minCapacity;
}

判断elementData是否为空数组,如果是空数组返回初始容量和size+1两个数中大的那个,这里应该返回初始容量10。
下个方法是,

private void ensureExplicitCapacity(int minCapacity) {
modCount++; // overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}

如果minCapacity大于数组的长度则进行扩容,

private void grow(int minCapacity) {
// overflow-conscious code
int oldCapacity = elementData.length;
int newCapacity = oldCapacity + (oldCapacity >> 1);
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);
}

上面代码的意思是计算原数组的大小并且加上其右移1位(原大小的1.5倍)记为newCapacity。把要扩容到的大小和newCapacity比较,取两者中比较大的那个作为数组的容量。把原先的数组中的元素拷贝到新的数组中。
在回头来看add方法中的elementData[size++] = e;由于已经对数组进行了扩容,则可以把e放在size+1的位置上,返回true。

下面看add(int index,E e)

在index处插入e元素,

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

验证index的合法性

private void rangeCheckForAdd(int index) {
if (index > size || index < 0)
throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
}

然后确定底层数组的容量,判断是否需要扩容,如果不需要则把e插入到index的位置,这里不是使用的循环移动插入,而是进行了数组的分部分拷贝,把index前的元素和index后的元素进行拷贝,最后把e插入在index处,这种方式比循环移动插入要快(使用底层的拷贝技术)。

2、get方法

根据index取得ArrayList中的元素,

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

判断给定的index的值比较和size的大小,如果超过了size则抛出异常,否则返回其index处的值。
3、contains(Object o)方法

判断o是否在ArrayList中,由于ArrayList使用数组存储元素,所以它是允许存重复值的,而contains方法返回的则是o第一次出现的位置。

public boolean contains(Object o) {
return indexOf(o) >= 0;
}
public int indexOf(Object o) {
if (o == null) {
for (int i = 0; i < size; i++)
if (elementData[i]==null)
return i;
} else {
for (int i = 0; i < size; i++)
if (o.equals(elementData[i]))
return i;
}
return -1;
}

可以看到ArrayList中允许存null值,且可以存多个。
如果o不是null,则从头开始循环直到第一个为o的元素出现,返回其下标否则返回-1,最后根据indexOf的值大于0的返回true,否则false。

4、remove(int index)/remove(Object o)

从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);
elementData[--size] = null; // clear to let GC do its work return oldValue;
}

首先依然是判断删除位置是否合法。如果合法则取得要删除索引位置的元素值(返回),因为要删除那么必然会进行数组的移动,即把删除位置后的数组向前移动,这里使用的是数组的拷贝。
remove(Object o)

按照元素进行删除,删除ArrayList中的第一个为o的元素,

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

这里仍然把null的情况单独处理,我们看不为null的情况的删除,使用size遍历整个数组,在数组中找到等于给定元素的下标,调用fastMove方法按照下标删除,只会把o第一次出现的位置上的元素删除,即重复的不会删除。

private void fastRemove(int index) {
modCount++;
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
}

上面还是按照下标进行删除。

未完待续。

java.util之一:ArrayList的更多相关文章

  1. java.util.ArrayList与java.util.Arrays$ArrayList区别

    本博客转载自:https://blog.csdn.net/maywehe/article/details/52553954 写demo的时候,为了避免用list.add方法,特意写了个数组然后转换成l ...

  2. java.util.Arrays$ArrayList addAll报错

    执行下面代码时报错: List<String> centerList = WebConstants.SUPPORT_BIG_CENTERS_LIST; // WebConstants.SU ...

  3. java.lang.ClassCastException: java.util.Arrays$ArrayList cannot be cast to java.util.ArrayList

    String[] 转换成 ArrayList 报的错. String[] str = {"A","B"}; ArrayList<String> li ...

  4. 关于 mybatis 报invalid comparison: java.util.Arrays$ArrayList and java.lang.String异常

    今天碰到个问题,来记录下,希望可以帮助到大家 贴错误源码: 这是一个根据list集合的查找数据的 sql,在接收list的时候加了判断 list != ‘ ’ “”,引起了集合与Stirng类型的比较 ...

  5. mybatis中传集合时 报异常 invalid comparison: java.util.Arrays$ArrayList and java.lang.String

    犯了一个低级的错误,在传集合类型的参数时,把他当成字符串处理了,导致报类型转换的错误 把  and nsrsbh!=' ' 删掉就行了

  6. Java-API:java.util.ArrayList

    ylbtech-Java-API:java.util.ArrayList 1.返回顶部   2.返回顶部   3.返回顶部   4.返回顶部   5.返回顶部 0. https://docs.orac ...

  7. JDK1.8源码(四)——java.util.Arrays 类

    java.util.Arrays 类是 JDK 提供的一个工具类,用来处理数组的各种方法,而且每个方法基本上都是静态方法,能直接通过类名Arrays调用. 1.asList public static ...

  8. java集合之ArrayList源码解读

    源自:jdk1.8.0_121 ArrayList继承自AbstractList,实现了List.RandomAccess.Cloneable.Serializable. ArrayList内部是通过 ...

  9. Java LinkedList 和 ArrayList

    Java 手册 java.util 类 ArrayList<E> java.lang.Object java.util.AbstractCollection<E> java.u ...

  10. java.lang.UnsupportedOperationException at java.util.AbstractList

    常常使用Arrays.asLisvt()后调用add,remove这些method时出现 Java.lang.UnsupportedOperationException异常.这是由于: Arrays. ...

随机推荐

  1. There is no getter for property named 'PRODUCT_ID' in 'class java.lang.String'

    背景:心血来潮之际,准备搭建以下多月未曾碰触过的Mybatis框架,体验一番原生之美.殊不知能力有限,错误百出.特抒此文以纪念此坑.问题:想在sql管理中来统一处理一些简单的判断,但是添加判断之后参数 ...

  2. 用js刷剑指offer(把数组排成最小的数)

    题目描述 输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个.例如输入数组{3,32,321},则打印出这三个数字能排成的最小数字为321323. 思路 对ve ...

  3. Java实现数据批量导入mysql数据库

    本文完全照搬别人的. 原文标题:Java实现数据批量导入数据库(优化速度-2种方法) 原文地址:https://blog.csdn.net/qy20115549/article/details/526 ...

  4. linux 非root用户安装nginx

    第一步:首先下载依赖包 下载地址 pcre(www.pcre.org),zlib(www.zlib.org),openssl(www.openssl.org) 第二步:上传那个nginx的安装包 下载 ...

  5. 【ocata】计算节点的 neutron 报错

    挂起/关闭实例时,/var/log/neutron/linuxbridge-agent.log中报错,但不影响使用 -- :: INFO neutron.plugins.ml2.drivers.age ...

  6. pyspark minHash LSH 查找相似度

    先看看官方文档: MinHash for Jaccard Distance MinHash is an LSH family for Jaccard distance where input feat ...

  7. IDEA -01 -忽略指定文件夹 -防止加载Vue-cli执行"npm install"命令后的项目时卡死

    问题描述 Vue的"npm install" 命令执行后,会生成一个很大的目录层次的"node_modules",文件十分繁多; idea加载这个项目下的文件夹 ...

  8. JDK源码那些事儿之ConcurrentLinkedQueue

    阻塞队列的实现前面已经讲解完毕,今天我们继续了解源码中非阻塞队列的实现,接下来就看一看ConcurrentLinkedQueue非阻塞队列是怎么完成操作的 前言 JDK版本号:1.8.0_171 Co ...

  9. 使用Nuget生成类库

     背景: 为了开始ITOO项目,我们先学习了一些基本知识,这些就是一个基本知识,这实现的是一种封装的思想,将方法打包,这样我们就可以在其他系统中直接引用而需要再耗费人力和财力去做,真正实现了高效率 ...

  10. Oracle 按指定顺序排序

    select * from (select 'Nick' as item from dual union all select 'Viki' as item from dual union all s ...