ArrayList 源码(基于Java1.8)
ArrayList 源码
ArrayList 基于数组实现,也就是类对变量 Object[]系列操作,封装为常用的add,remove,indexOf, contains本质是通过 size 计数器对数组进行系列的操作实现
add 方法和 toArray 也是借助 Arrays 工具类完成
还有很有意思的地方就是add的扩容方法 ensureCapacityInternal() 就会增加50%容量,
因为底层数组是不可变的,所以add实际上是通过Arrays工具再拷贝扩容后长度的数组,然后覆盖当前数组
预设集合长度是最优的操作,今天看源码才明白原理,ArrayList默认构造器为内部数组定义的长度为10,当list超出指定集合的时候,ArrayList会操作自动扩容,
扩容的方法是使用Arrays.copyOf方法重新构造新的数组,然后将旧数组的数据全部拷贝到新的数组里面,重新生成,再拷贝数组,消耗是巨大的,这也导致了 ArrayList 天生的缺点,插入删除元素慢
源码中可以看到,ArrayList继承抽象类AbstractList,实现List接口,RandomAccess, Cloneable, Serializable等接口主要用于标识,并没有很大作用
DEFAULT_CAPACITY = 10 是集合的默认初始化长度,我们常用new ArrayList<>()构造,其实就是创建一个长度为10的Object[]数组
size 为计数器,记录当前数组的长度
构造器的代码中,看下最常用的两个构造器,带参数initialCapacity的构造器,直接就让变量创建对应长度的数组,如果initialCapacity == 0 就直接赋值为空数组 Object[] elementData = {};
下面无参构造器,就直接初始化容量为10的空列表
上面的自动扩展数组容量的方法,每次添加元素都会调用该方法,确保集合的长度足够用,上面代码中最有趣的是2行
newCapacity = oldCapacity + (oldCapacity >> 1), 既每次扩大容量为当前容量的50%,这里的扩容是固定的,和HashMap有负载因子不同
然后调用 Arrays.copyOf() 将旧数组内容和新长度作为参数,创建新的Object[]数组,并且将原数组的元素,拷贝到新的数组里面
因为数组长度是不可变的,所以这里的做法是,创建新的数组,再拷贝元素,这里也是ArrayList性能最低的地方
常用的size,isEmpty,contains,indexOf 也只是对Object[] 进行系列的循环判断操作
get方法,也是指通过下标,从Object[]数组中获取对象,然后转型 Object -> E
add末尾添加元素方法步骤,先判断容量是否够用,然后在数组末尾elementData[size++] == e,加入新的元素,size为数组长度
remove 方法,也是通过循环equals匹配,找到下标,然后通过下标删除,—size,减去计数器长度,并没有实际减少数组长度
clear方法也相同,把数组的值都设为null,然后 size = 0;
总结:
通过以上对ArrayList关键部分源码分析,通晓ArrayList原理,我们可以得出结论:
- arrayList 底层来自数组,通过下标随机访问元素,速度非常快,但是插入和删除,要频繁的移动数组元素,改变数组长度,效率比较低
- arrayList 主要性能问题在于当数组元素达到容量上限,扩容的成本很大,所以能够预先知道元素,然后使用 new ArrayList(int initialCapacity) 手动设置容量是最好的
ArrayList 源码(基于Java1.8)的更多相关文章
- 死磕Java之聊聊ArrayList源码(基于JDK1.8)
工作快一年了,近期打算研究一下JDK的源码,也就因此有了死磕java系列 ArrayList 是一个数组队列,相当于动态数组.与Java中的数组相比,它的容量能动态增长.它继承于AbstractLis ...
- Java集合(四)--基于JDK1.8的ArrayList源码解读
public class ArrayList<E> extends AbstractList<E> implements List<E>, RandomAccess ...
- Java -- 基于JDK1.8的ArrayList源码分析
1,前言 很久没有写博客了,很想念大家,18年都快过完了,才开始写第一篇,争取后面每周写点,权当是记录,因为最近在看JDK的Collection,而且ArrayList源码这一块也经常被面试官问道,所 ...
- java8 ArrayList源码阅读
转载自 java8 ArrayList源码阅读 本文基于jdk1.8 JavaCollection库中有三类:List,Queue,Set 其中List,有三个子实现类:ArrayList,Vecto ...
- ArrayList源码解析
ArrayList简介 ArrayList定义 1 public class ArrayList<E> extends AbstractList<E> implements L ...
- ArrayList源码剖析
ArrayList简介 ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长,类似于C语言中的动态申请内存,动态增长内存. ArrayList不是线程安全的,只能用在单线程环境下,多线 ...
- 转:【Java集合源码剖析】ArrayList源码剖析
转载请注明出处:http://blog.csdn.net/ns_code/article/details/35568011 本篇博文参加了CSDN博文大赛,如果您觉得这篇博文不错,希望您能帮我投一 ...
- ArrayList源码学习----JDK1.7
什么是ArrayList? ArrayList是存储一组数据的集合,底层也是基于数组的方式实现,实际上也是对数组元素的增删改查:它的主要特点是: 有序:(基于数组实现) 随机访问速度快:(进行随机访问 ...
- ArrayList源码分析超详细
ArrayList源码分析超详解 想要分析下源码是件好事,但是如何去进行分析呢?以我的例子来说,我进行源码分析的过程如下几步: 找到类:利用 IDEA 找到所需要分析的类(ztrl+N查找ArraLi ...
随机推荐
- (转)Unity3D中移动物体位置的几种方法
1. 简介 在unity3d中,有多种方式可以改变物体的坐标,实现移动的目的,其本质是每帧修改物体的position. 2. 通过Transform组件移动物体 Transform 组件用于描述物体在 ...
- Linux的硬盘使用情况、挂载、SSD挂载(查看df -h不能看到的卷)
linux上的盘和window的有区别,磁盘空间必须挂载在目录上,要不然没用 对与新增的硬盘.SSD固态硬盘.挂载到linux上的操作如下: df -h #显示目前在Linux系统上的文件系 ...
- 深入理解计算机系统chapter2
---恢复内容开始--- 整数表示: 反码和原码都会有正零和负零 有符号整数和无符号整数之间的转换 反之 扩展一个数字的位级表示 截断操作 无符号加法的益处 补码的加法 规格化的值:E=e-bias ...
- 9月24日noip模拟赛解题报告
1.校门外的树(tree.c/cpp/pas 128M,1s) Description LSGJ扩建了,于是校门外有了一条长为L的路.路上种了一排的树,每相邻两棵树之间的距离为1,我们可以把马路看成一 ...
- Java 数组扩容
在添加数据到达数组的上限的时候数组进行扩容: public void resizeArrayCaptcity(){ if(size>=arr.length){ Emp [] arr2=new ...
- Pro Flight YOKE 设备键位映射踩过的坑
背景 VR游戏项目.街机游戏项目7月阶段版本快要结束了,考虑到带有键鼠外设显得逼格比较Low,所以决定采用"高大上"的专业设备来进行游戏操作. 需求 需要将键盘鼠标操作的18个键位 ...
- 使用paramiko远程执行
写部署脚本时,难免涉及到一些远程执行命令或者传输文件. 之前一直使用sh库,调用sh.ssh远程执行一些命令,sh.scp传输文件,但是实际使用中还是比较麻烦的,光是模拟用户登陆这一点,还需要单独定义 ...
- Clojure——学习迷宫生成
背景 初学clojure,想着看一些算法来熟悉clojure语法及相关算法实现. 找到一个各种语言生成迷宫的网站:http://rosettacode.org/wiki/Maze_generation ...
- 偏置-方差分解(Bias-Variance Decomposition)
本文地址为:http://www.cnblogs.com/kemaswill/,作者联系方式为kemaswill@163.com,转载请注明出处. 机器学习的目标是学得一个泛化能力比较好的模型.所谓泛 ...
- MySQL or MariaDB 错误解决方法之报错代码1045
phpMyAdmin登录报错:mysqli_real_connect(): (28000/1045): Access denied for user 'root'@'localhost' (using ...