ArrayList 扩容详解,扩容原理

ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长。
ArrayList不是线程安全的,只能用在单线程环境下。
实现了Serializable接口,因此它支持序列化,能够通过序列化传输;
实现了RandomAccess接口,支持快速随机访问,实际上就是通过下标序号进行快速访问;
实现了Cloneable接口,能被克隆。

动态扩容

一 初始化

首先有三种方式来初始化:

public ArrayList();

默认的构造器,将会以默认的大小来初始化内部的数组

public ArrayList(Collection<? extends E> c)

用一个ICollection对象来构造,并将该集合的元素添加到ArrayList

public ArrayList(int initialCapacity)

用指定的大小来初始化内部的数组

后两种方式都可以理解,通过创造对象,或指定大小来初始化内部数据即可。
那我们来重点关注一下无参数构造器的实现过程:

/*** Constructs an empty list with an initial capacity of ten.*/public ArrayList() {// DEFAULTCAPACITY_EMPTY_ELEMENTDATA是空数组this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;}private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};

可以看出它的默认数组为长度为0。而在之前JDK1,6中,无参数构造器代码是初始长度为10。
JDK6代码这样的:

public ArrayList() {this(10);}public ArrayList(int initialCapacity) {super();if (initialCapacity < 0)throw new IllegalArgumentException("Illegal Capacity: "+initialCapacity);this.elementData = new Object[initialCapacity];}

接下来,要扩容的话,肯定是在ArrayList.add 方法中。我们来看一下具体实现。

二 确保内部容量

我们以无参数构造为例,
初始化时,数组长度为0.
那我现在要添加数据了,数组的长度是怎么变化的?

public boolean add(E e) {//确保内部容量(通过判断,如果够则不进行操作;容量不够就扩容来确保内部容量)ensureCapacityInternal(size + 1);  // ①Increments modCount!!elementData[size++] = e;//②return true;}

① ensureCapacityInternal方法名的英文大致是“确保内部容量”,size表示的是执行添加之前的元素个数,并非ArrayList的容量,容量应该是数组elementData的长度。ensureCapacityInternal该方法通过将现有的元素个数数组的容量比较。看如果需要扩容,则扩容。
②是将要添加的元素放置到相应的数组中。
下面具体看 ensureCapacityInternal(size + 1);

// ① 是如何判断和扩容的。
private void ensureCapacityInternal(int minCapacity) {//如果实际存储数组 是空数组,则最小需要容量就是默认容量if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);}ensureExplicitCapacity(minCapacity);}private void ensureExplicitCapacity(int minCapacity) {modCount++;//如果数组(elementData)的长度小于最小需要的容量(minCapacity)就扩容if (minCapacity - elementData.length > 0)grow(minCapacity);}/*** Default initial capacity.*/private static final int DEFAULT_CAPACITY = 10;

以上,elementData是用来存储实际内容的数组。minExpand 是最小扩充容量。
DEFAULTCAPACITY_EMPTY_ELEMENTDATA共享的空数组实例用于默认大小的空实例。根据传入的最小需要容量minCapacity来和数组的容量长度对比,若minCapactity大于或等于数组容量,则需要进行扩容。

三 扩容

/**增加容量,以确保它至少能容纳*由最小容量参数指定的元素数。* @param mincapacity所需的最小容量*/private void grow(int minCapacity) {// overflow-conscious codeint oldCapacity = elementData.length;//>>位运算,右移动一位。 整体相当于newCapacity =oldCapacity + 0.5 * oldCapacity  // jdk1.7采用位运算比以前的计算方式更快int newCapacity = oldCapacity + (oldCapacity >> 1);if (newCapacity - minCapacity < 0)newCapacity = minCapacity;//jdk1.7这里增加了对元素个数的最大个数判断,jdk1.7以前是没有最大值判断的,MAX_ARRAY_SIZE 为int最大值减去8(不清楚为什么用这个值做比较)if (newCapacity - MAX_ARRAY_SIZE > 0)newCapacity = hugeCapacity(minCapacity);// 最重要的复制元素方法elementData = Arrays.copyOf(elementData, newCapacity);}

综上所述ArrayList相当于在没指定initialCapacity时就是会使用延迟分配对象数组空间,当第一次插入元素时才分配10(默认)个对象空间。假如有20个数据需要添加,那么会分别在第一次的时候,将ArrayList的容量变为10 (如下图一);之后扩容会按照1.5倍增长。也就是当添加第11个数据的时候,Arraylist继续扩容变为10*1.5=15(如下图二);当添加第16个数据时,继续扩容变为15 * 1.5 =22个(如下图四)。:

**  向数组中添加第一个元素时,数组容量为10.**

**  向数组中添加到第10个元素时,数组容量仍为10.**

**  向数组中添加到第11个元素时,数组容量扩为15.**

**  向数组中添加到第16个元素时,数组容量扩为22.**

每次扩容都是通过Arrays.copyOf(elementData, newCapacity) 这样的方式实现的。

**  对比和总结:**

本文介绍了 ArrayList动态扩容的全过程。如果通过无参构造的话,初始数组容量为0,当真正对数组进行添加时,才真正分配容量。每次按照1.5倍(位运算)的比率通过copeOf的方式扩容。 在JKD1.6中实现是,如果通过无参构造的话,初始数组容量为10,每次通过copeOf的方式扩容后容量为原来的1.5倍,以上就是动态扩容的原理。

ArrayList 扩容详解,扩容原理相关推荐

  1. 云服务器存储扩容详解

    云服务器存储扩容详解 2020-01-20阅读 6640 环境说明 Linux操作系统:CentOS Linux release 7.6.1810 (Core) 平台环境:腾讯云CVM,CBS 本教程 ...

  2. Android 4.4 NotificationManagerService使用详解与原理分析(二)__原理分析

    前置文章: <Android 4.4 KitKat NotificationManagerService使用详解与原理分析(一)__使用详解> 转载请务必注明出处:http://blog. ...

  3. 自动驾驶算法-滤波器系列(一)——详解卡尔曼滤波原理

    详解卡尔曼滤波原理 什么是卡尔曼滤波? 我们能用卡尔曼滤波做什么? 卡尔曼滤波是如何看到你的问题的 使用矩阵来描述问题 外部控制量 外部干扰 用测量量来修正估计值 融合高斯分布 将所有公式整合起来 总 ...

  4. Batch Normalization详解(原理+实验分析)

    Batch Normalization详解(原理+实验分析) 1. 计算过程 2. 前向传播过程 3. 反向传播过程 4. 实验分析 4.1 实验一:验证有没有BatchNorm下准确率的区别 4.2 ...

  5. P2P技术详解(一):NAT详解——详细原理、P2P简介(转)

    这是一篇介绍NAT技术要点的精华文章,来自华3通信官方资料库,文中对NAT技术原理的介绍很全面也很权威,对网络应用的应用层开发人员而言有很高的参考价值. <P2P技术详解>系列文章 ➊ 本 ...

  6. 015. P2P技术详解(一):NAT详解——详细原理、P2P简介

    http://www.52im.net/thread-50-1-1.html 这是一篇介绍NAT技术要点的精华文章,来自华3通信官方资料库,文中对NAT技术原理的介绍很全面也很权威,对网络应用的应用层 ...

  7. 原码,反码,补码详解及原理

    原码,反码,补码详解及原理 原码  原码:在数值前直接加一符号位的表示法. 例如: 符号位 数值位 byte的取值范围 [+7]原= 0 0000111 B[-7]原= 1 0000111 B 注意: ...

  8. 新书推荐 |《5G NR物理层技术详解:原理、模型和组件》

    新书推荐 <5G NR物理层技术详解:原理.模型和组件> 点击上图了解及购买 5G专家和学者撰写,详解5G NR物理层技术(包括波形.编码调制.信道仿真和多天线技术等)及其背后的成因,爱立 ...

  9. 快速排序算法详解(原理,时间复杂度,实现代码)

    快速排序算法详解(原理.实现和时间复杂度) 快速排序是对冒泡排序的一种改进,由 C.A.R.Hoare(Charles Antony Richard Hoare,东尼·霍尔)在 1962 年提出. 快 ...

最新文章

  1. Java多线程并发常用类实例之:exchanger
  2. Android中解决debug.keystore到期的问题
  3. java 反射 工厂_JAVA反射机制、工厂模式与SPRING IOC
  4. Bootstrap CustomBox 弹层
  5. AD管控下的弹性云桌面和文件共享最佳实践
  6. ASP.NET Core 源码学习之 Logging[2]:Configure
  7. python入门:输出1-100之内的所有奇数和偶数
  8. 房间匹配机制--自走旗
  9. Python将.nii格式文件转换为.png格式
  10. React中文文档之Lifting State Up
  11. 2018最新安卓面试大全(含BAT,网易,滴滴)----你面不上BAT的原因:面经宝典,都在这里啦
  12. python中一切内容都可以称为_Python中对象的概念很广泛,Python中的一切内容都可以称为 。...
  13. 9个offer,12家公司
  14. PHP计算两个日期相差的天数方法详解
  15. 计算机故障报警声2声,电脑报警声含义大全 主板警报声判断故障
  16. iBatis 动态SQL别名查询结果为Null问题
  17. 2022-2027年中国BOSS系统行业市场全景评估及发展战略规划报告
  18. 电子科技大学计算机考研招生人数,电子科技大学2020年考研报名人数统计
  19. h5实现苹果悬浮球_原来不止苹果手机有悬浮球,华为手机也能设置,厉害了!...
  20. 玩转Redis集群之Sentinel

热门文章

  1. iOS各种系统服务的代码调用
  2. 武大计算机博士好毕业吗,2021武大辅导员拟录名单公布,博士生占多数,大多为名校毕业...
  3. 基于RGB-D的6D目标检测算法
  4. 使用Arthas排查解决项目部署后CPU飙高问题
  5. Hosts文件是做什么的
  6. 退休计划失败,64岁高龄程序员重新开始工作
  7. 科技的成就(四十八)
  8. Android 数据储存的方式之本地数据库储存
  9. MFC Windows 程序设计(5)
  10. canvas字体样式