Java的ArrayList扩容源码解析

 更新时间:2024年01月11日 10:02:56   作者:好奇的7号  
这篇文章主要介绍了Java的ArrayList扩容源码解析,通过动态扩容,ArrayList能够在添加元素时保持高效的性能,扩容操作是有一定开销的,但由于扩容的时间复杂度为O(n),其中n是当前元素个数,所以平均情况下,每次添加元素的时间复杂度仍然是O(1),需要的朋友可以参考下

ArrayList扩容源码

1、扩容原理概括

ArrayList的扩容原理如下:

  • 初始容量:在创建ArrayList对象时,默认会分配一个初始容量。初始容量可以通过构造函数进行指定,若未指定,则默认为10。例如,ArrayList<String> list = new ArrayList<>();会创建一个初始容量为10的ArrayList。
  • 添加元素:当向ArrayList中添加元素时,会先检查当前元素个数是否已经达到了数组的容量上限。如果元素个数等于或超过了容量上限,就需要进行扩容。
  • 扩容机制:扩容时,ArrayList会创建一个更大的新数组,并将原有的元素复制到新数组中。默认情况下,新数组的大小是原来容量的1.5倍(即增长50%)。例如,如果当前容量为10,那么扩容后的新容量为15。
  • 数组复制:在扩容时,ArrayList使用System.arraycopy()方法将原始数组中的元素复制到新的数组中。这是一个底层的高效数组复制方法,可以快速地将原有的数据移动到新数组中。
  • 更新引用:在完成数组复制后,ArrayList会更新内部的引用,指向新的数组。这样,原先的数组就会被垃圾回收器回收。

通过动态扩容,ArrayList能够在添加元素时保持高效的性能。扩容操作是有一定开销的,但由于扩容的时间复杂度为O(n),其中n是当前元素个数,所以平均情况下,每次添加元素的时间复杂度仍然是O(1)。

同时,扩容的增长因子(即容量的增加比例)也可以被修改,以满足特定需求。

2、源码解析

例如,对于如下list进行添加元素,初始容量设置为5,添加6个元素:

ArrayList<String> list = new ArrayList<>(5);
list.add("1");
list.add("2");
list.add("3");
list.add("777");
list.add("888");
list.add("999");

对于一开始的前五个元素而言:

public boolean add(E e) {
        modCount++;
        add(e, elementData, size);//e是“1”,size是0,即目前元素数量,也可以理解为数组的索引
        return true;
}
 
//add源码:
private void add(E e, Object[] elementData, int s) {
        if (s == elementData.length)//当元素数量等于arraylist数组长度,才grow扩容
            elementData = grow();
        elementData[s] = e;//否则直接在索引位置赋值
        size = s + 1;//并将索引右移
}

但在第6个元素“999”添加进去的时候,要进入grow方法进行扩容:

private void add(E e, Object[] elementData, int s) { //e是“999”,s为5
        if (s == elementData.length)//已经相等
            elementData = grow();//步入grow
        elementData[s] = e;
        size = s + 1;
}
 
//grow源码:输入size + 1
private Object[] grow(int minCapacity) {
        int oldCapacity = elementData.length;//旧的数组大小
 
//接下来,检查当前数组是否为空,或者不是使用默认初始容量的空数组(即DEFAULTCAPACITY_EMPTY_ELEMENTDATA)。如果是空数组,则说明是第一次添加元素,使用默认初始容量。
        if (oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            int newCapacity = ArraysSupport.newLength(oldCapacity,
                    minCapacity - oldCapacity, /* minimum growth */
                    oldCapacity >> 1           /* preferred growth */);//计算新数组大小
            return elementData = Arrays.copyOf(elementData, newCapacity);
        } else {
            return elementData = new Object[Math.max(DEFAULT_CAPACITY, minCapacity)];
        }
}
 
//ArraysSupport.newLength源码:
public static int newLength(int oldLength, int minGrowth, int prefGrowth) {
        // assert oldLength >= 0
        // assert minGrowth > 0
//minGrowth就是6 - 5 = 1,即,至少需要增大的容量
//prefGrowth为数组的旧容量的1/2
//求其max,一般为1/2。加上oldlength,所以为1.5倍
        int newLength = Math.max(minGrowth, prefGrowth) + oldLength;
        if (newLength - MAX_ARRAY_LENGTH <= 0) {
            return newLength;
        }
        return hugeLength(oldLength, minGrowth);
    }
 
//最后调用Arrays.copyOf,内部使用System.arraycopy()方法将原始数组中的元素复制到新的数组中

实现扩容后,添加元素步骤同上,结束,return true

到此这篇关于Java的ArrayList扩容源码解析的文章就介绍到这了,更多相关ArrayList扩容源码内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Mybatis执行SQL命令的流程分析

    Mybatis执行SQL命令的流程分析

    这篇文章主要介绍了Mybatis执行SQL命令的流程分析,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-04-04
  • 详解Spring Boot对 Apache Pulsar的支持

    详解Spring Boot对 Apache Pulsar的支持

    Spring Boot通过提供spring-pulsar和spring-pulsar-reactive自动配置支持Apache Pulsar,类路径中这些依赖存在时,Spring Boot自动配置命令式和反应式Pulsar组件,PulsarClient自动注册,默认连接本地Pulsar实例,感兴趣的朋友一起看看吧
    2024-11-11
  • java简单坦克大战制作代码

    java简单坦克大战制作代码

    这篇文章主要介绍了java简单坦克大战制作代码,利用Java语言中的集合、Swing、线程等知识点编写一个坦克大战游戏,需要的朋友可以参考下
    2016-07-07
  • SpringBoot详解整合MyBatis过程中可能遇到的问题

    SpringBoot详解整合MyBatis过程中可能遇到的问题

    因为Spring Boot框架开发的便利性,所以实现Spring Boot与数据访问层框架(例如MyBatis)的整合非常简单,主要是引入对应的依赖启动器,并进行数据库相关参数设置即可
    2022-07-07
  • SpringBoot中注解@ConfigurationProperties与@Value的区别与使用详解

    SpringBoot中注解@ConfigurationProperties与@Value的区别与使用详解

    本文主要介绍了SpringBoot中注解@ConfigurationProperties与@Value的区别与使用,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09
  • 深入理解Java中的volatile关键字(总结篇)

    深入理解Java中的volatile关键字(总结篇)

    volatile这个关键字,不仅仅在Java语言中有,在很多语言中都有的,而且其用法和语义也都是不尽相同的。这篇文章主要介绍了Java中的volatile关键字,需要的朋友可以参考下
    2018-10-10
  • Java核心库实现简单的AOP

    Java核心库实现简单的AOP

    这篇文章主要介绍了如何用Java核心库实现简单的AOP,帮助大家为了更好的理解和学习AOP的思想,感兴趣的朋友可以了解下
    2020-08-08
  • 详解Java 微服务架构

    详解Java 微服务架构

    这篇文章主要介绍了Java 微服务架构的相关资料,帮助大家更好的理解和使用Java,感兴趣的朋友可以了解下
    2021-02-02
  • Java8新特性Lambda表达式的一些复杂用法总结

    Java8新特性Lambda表达式的一些复杂用法总结

    lambda表达式是JAVA8中提供的一种新的特性,它支持Java也能进行简单的“函数式编程”。 下面这篇文章主要给大家介绍了关于Java8新特性Lambda表达式的一些复杂用法的相关资料,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-07-07
  • Java 代码本地设置Hadoop用户名密码的方法

    Java 代码本地设置Hadoop用户名密码的方法

    在Hadoop环境中,通常使用Kerberos进行身份验证,这篇文章主要介绍了Java 代码本地设置Hadoop用户名密码的方法,需要的朋友可以参考下
    2024-08-08

最新评论