前言

本文主要给大家介绍了关于Java中Arraylist动态扩容的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧。
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 code
int 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倍,以上就是动态扩容的原理。
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。
参考资料:http://blog.csdn.net/u010176014/article/details/52073339
# java
# arraylist
# 扩容
# arraylist动态扩容
# java中的arraylist
# 关于Java的ArrayList数组自动扩容机制
# Java ArrayList扩容机制原理深入分析
# Java中的ArrayList容量及扩容方式
# Java基础之ArrayList的扩容机制
# 在java中ArrayList集合底层的扩容原理
# Java使用数组实现ArrayList的动态扩容的方法
# 对Java ArrayList的自动扩容机制示例讲解
# Java ArrayList扩容问题实例详解
# 浅谈Java中ArrayList的扩容机制
# 组中
# 到第
# 长度为
# 实现了
# 的是
# 都是
# 是一个
# 是在
# 序列化
# 好了
# 相关内容
# 第一个
# 说了
# 是怎么
# 最重要
# 两种
# 不多
# 而在
# 所需
# 不清楚
相关文章:
一键网站制作软件,义乌购一件代发流程?
金*站制作公司有哪些,金华教育集团官网?
如何选购建站域名与空间?自助平台全解析
音乐网站服务器如何优化API响应速度?
JS中使用new Date(str)创建时间对象不兼容firefox和ie的解决方法(两种)
移动端手机网站制作软件,掌上时代,移动端网站的谷歌SEO该如何做?
Swift开发中switch语句值绑定模式
浙江网站制作公司有哪些,浙江栢塑信息技术有限公司定制网站做的怎么样?
网站专业制作公司有哪些,做一个公司网站要多少钱?
如何在沈阳梯子盘古建站优化SEO排名与功能模块?
制作网站的软件免费下载,免费制作app哪个平台好?
如何在香港服务器上快速搭建免备案网站?
广州网站制作的公司,现在专门做网站的公司有没有哪几家是比较好的,性价比高,模板也多的?
宁波自助建站系统如何快速打造专业企业网站?
C++用Dijkstra(迪杰斯特拉)算法求最短路径
威客平台建站流程解析:高效搭建教程与设计优化方案
建站与域名管理如何高效结合?
网站制作公司,橙子建站是合法的吗?
云南网站制作公司有哪些,云南最好的招聘网站是哪个?
制作证书网站有哪些,全国城建培训中心证书查询官网?
,制作一个手机app网站要多少钱?
在线流程图制作网站手机版,谁能推荐几个好的CG原画资源网站么?
如何选择高效便捷的WAP商城建站系统?
常州企业网站制作公司,全国继续教育网怎么登录?
怎么制作一个起泡网,水泡粪全漏粪育肥舍冬季氨气超过25ppm,可以有哪些措施降低舍内氨气水平?
内部网站制作流程,如何建立公司内部网站?
阿里云高弹*务器配置方案|支持分布式架构与多节点部署
如何在建站之星网店版论坛获取技术支持?
详解ASP.NET 生成二维码实例(采用ThoughtWorks.QRCode和QrCode.Net两种方式)
导航网站建站方案与优化指南:一站式高效搭建技巧解析
c# 在高并发场景下,委托和接口调用的性能对比
图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?
如何使用Golang table-driven基准测试_多组数据测量函数效率
网站制作免费,什么网站能看正片电影?
C#如何在一个XML文件中查找并替换文本内容
再谈Python中的字符串与字符编码(推荐)
湖南网站制作公司,湖南上善若水科技有限公司做什么的?
手机网站制作平台,手机靓号代理商怎么制作属于自己的手机靓号网站?
小程序网站制作需要准备什么资料,如何制作小程序?
如何在云主机上快速搭建多站点网站?
公司门户网站制作流程,华为官网怎么做?
厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?
上海网站制作网页,上海本地的生活网站有哪些?最好包括生活的各个方面的?
如何通过IIS搭建网站并配置访问权限?
建站之星安装后如何配置SEO及设计样式?
建站之星展会模板:智能建站与自助搭建高效解决方案
孙琪峥织梦建站教程如何优化数据库安全?
青浦网站制作公司有哪些,苹果官网发货地是哪里?
如何在IIS服务器上快速部署高效网站?
如何选择美橙互联多站合一建站方案?
*请认真填写需求信息,我们会在24小时内与您取得联系。