Java集合源码之ArrayList

  便于增强自己对集合的理解和记忆,准备出一系列java集合源码的阅读以及解析,本系列基于JDK1.8。

集合框架是java中最基础也是最重要的一个学习模块之一,因为工作中每天都要用,如果对于集合总是一知半解,或者只知其然不知其所以然,这里我指的是需要“背的面试题”,只记得某个集合的特性,但不知其如何实现。

我认为阅读源码主要分三步:
  • 阅读改类或接口的继承与实现关系,也就是看类图。
  • 阅读构造方法,可以看到他具备哪些重要的构造参数。
  • 阅读重要方法,分析实现原理。

一、ArrayList简介

想必大家对ArrayList并不陌生,甚至老生常谈的一个集合类。确实,他是我们在集合中最经常使用的一个类。它是一个能动态增长的动态数组,相比于普通数组,它具备自动扩容的功能(具体如何扩容后面会有讲解)。

先来看看它直接继承和实现了哪些类和接口:
image.png

ok,从源码我们得知它直接继承了抽象类AbstractList,实现了List【具备增删改查元素功能】,RandomAccess【随机访问功能,是java中用来被List实现,为List提供快速访问功能的。在ArrayList中,我们即可以通过元素的序号快速获取元素对象;这就是快速随机访问。反例就是 LinkedList, 链表结构使得它不支持随机访问,只能按序访问,因此在一些操作上性能略逊一筹。】, Cloneable【可克隆,覆盖了函数clone(),能被克隆。】, java.io.Serializable【可序列化】接口。

来看看它的继承关系:

image.png

可以看出最顶层的接口是Collection,关于Collection可以参考大佬文章,写的非常生动详细:
https://blog.csdn.net/u011240877/article/details/52773577

本文只列举重要api,我们从List开始翻,它是Collection的子接口之一。它定义的方法有:

int size();
   返回List的长度。

boolean isEmpty();
   判断是否为空。

Object[] toArray();
   将List转为数组。

boolean add(E e);
   向尾部添加元素。

boolean remove(Object o);
   删除元素。

以上都是List继承自Collection的方法,下面是它自己扩展的方法:

int indexOf(Object o);
   返回指定元素第一次出现的索引,如果该列表不包含该元素,则为-1。

ListIterator<E> listIterator();
返回一个ListIterator ,Iterator和ListIterator的区别会在其他文章讲解。

List<E> subList(int fromIndex, int toIndex);
截取该List的一部分并返回一个新的List。但是值得注意的是,对subList后返回的引用操作也会对原来的List造成影响。请看:


image.png

结果是:


image.png

ArrayList的重要属性:

    /**
     * 默认的初始值为10
     */
    private static final int DEFAULT_CAPACITY = 10;
  /**
     * 一个空对象,用于直接赋值
     */
    private static final Object[] EMPTY_ELEMENTDATA = {};
  /**
     *一个空对象,如果使用默认构造函数创建,则默认对象内容默认是该值
     */
   private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
  /**
   * 用于存放数据,不参与序列化
  */
  transient Object[] elementData;
/**
*  可以分配的最大大小为 Integer.MAX_VALUE - 8
*/
    private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

    /**
     * 当前元素个数
     */
    private int size;

主要需要我们知道的就是它的初始值为10,然后采用的Object[ ]来存放数据。
还有一个参数叫modCount,根据源码注释得知他记录着List的结构被修改的次数!(这里的结构修改指add,remove操作,set不算)有的文章会说这个属性代表该List扩容过几次其实不然。

protected transient int modCount = 0;

ArrayList的构造方法:

image.png

1.这是个空构造函数,直接把默认的之前定义的空对象数组赋给该引用。

image.png

2.需要指定初始化容量的构造,如果初始值大于0就new一个初始值大小的对象数组。如果等于0使用空数组,小于0就会报异常了。

image.png

3.构造一个包含指定类或其子类的集合,参数为一个集合。
1)将参数集合转换为对象数组并将9引用赋予elementData。
2)更新size并判断是否为0,如果为0则赋为空数组,否则执行Array.copyOf方法。
3)将参数集合的元素内容进行深拷贝一份赋给elementData。因为在此之前elementData仅仅是指向了该参数集合。ps:有的同学可能不懂什么是深拷贝,浅拷贝,简单说明一下:浅拷贝就是将目标对象的引用拷贝一份 ,换句话说如果你修改了副本的一些引用字段那么原对象的引用字段也会被修改,反之深拷贝是将内容拷贝一份,副本无论怎么修改都不会影响原对象。(如果还是不明白请Google一下)

去除多余空间 trimToSize()

image.png

由于List在使用过程中会扩容,以至于有的空间会被浪费(因为数组是需要预先预定空间的),所以这个方法可以根据elementData重新copy出一份大小为当前elementData的size的对象数组并把元素也copy进去再赋值给elementData,达到节省空间的目的。这里可以看到modCount被自增了,所以应了我之前说的modCount不一定代表扩容了几次,而是list的结构被修改了几次。(增删都算)它可以用于在使用迭代器遍历集合的时候同时修改集合元素。详见大佬文章:https://www.cnblogs.com/zuochengsi-9/p/7050351.html
而且在使用迭代器遍历的时候,这个变量可以提醒我们是否线程安全。这个我会单独写一篇文章来说明。

添加元素 add(E e)

image.png
image.png

由此可知add先调用了ensureCapacityInternal(),然后调用了calculateCapacity()方法。第一次看我也不知道为什要有calculateCapacity()这个方法,直接判断不就好了吗?其实它在确保第一次添加的时候能有地方能存数据,因为如果我们可能会用空构造new一个ArrayList,会被赋值空数组。所以判断一下,如果是空数组需要在默认值与期望值之间选择大的一个,选择大的是因为扩容也需要消耗性能,所以一次扩大一点,省的可能还要多扩几次。否则返回size+1。

第三步,如果添加该元素后的元素个数大于了数组最大长度将会调用扩容。

look一下扩容方法grow():

image.png

由此看出,新的长度=旧长度(旧长度/2)。也就是原来的1.5倍!如果扩容后的长度还是小于期望长度则将期望长度赋给新长度。一般来说我们数组长度不会超过MAX_ARRAY_SIZE。最后拷贝一份给elementData。

总结:add主要的执行逻辑如下:
1)确保数组已使用长度加1之后足够存下 下一个数据
2)修改次数modCount 标识自增1,如果当前数组已使用长度(size)加1后的大于当前的数组长度,则调用grow方法,增长数组,grow方法会将当前数组的长度变为原来容量的1.5倍。
3)将新元素添加到位于size的位置上。
4)返回添加成功布尔值。

是否包含元素contains(Object o) :

image.png

里面就是调用了indexOf方法去遍历数组如果存在返回值大于0,否则-1。

返回数组中第一个匹配项的索引下标 indexOf(Object o):

image.png

copy一个数组并返回:

image.png

copy数组方法的重载,使用现有数组来接收:

image.png

如果参数数组小于元素个数就,创建一个新的容量大小为当前list元素个数的数组并把元素copy过去最后返回。否则直接copy进参数数组。

System.arraycopy 方法
src 原数组
srcPos 原数组起始位置
dest 目标数组
destPos 目标数组的起始位置
length 要复制的数组元素的数目

清除元素 clear():

image.png

将数组中的所有元素引用置为null等待GC回收。

获得指定索引的元素get():

image.png

首先检查index范围是否合法,否则报出越界异常。

set方法:

image.png

删除方法 remove(int index):

image.png

首先检查是否越界,将modcount+1;将index后面的元素都往前挪一位。将该元素引用置空,返回原来的oldValue。(System.arraycopy是浅拷贝)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351