Java中 ArrayList 的扩容机制深度解析

2024-08-24 21:04

本文主要是介绍Java中 ArrayList 的扩容机制深度解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在 Java 8 中,ArrayList 是一个广泛使用的集合类,它提供了动态数组的功能。当 ArrayList 需要容纳更多的元素时,它会自动进行扩容。本文将深入探讨 ArrayList 的扩容机制,特别是 grow 方法的实现细节。

ArrayList 扩容概述

ArrayList 通过一个叫做 elementData 的内部数组来存储元素。当添加元素导致数组满时,ArrayList 会执行扩容操作。扩容过程包括以下几个步骤:

  1. 计算新容量。
  2. 检查是否需要满足最小容量要求。
  3. 确保不超过 JVM 允许的最大数组大小。
  4. 复制旧数组到新数组。

grow 方法源码解析

grow 方法是扩容操作的核心。以下是 grow 方法的源码,以及对关键点的注释说明:

private void grow(int minCapacity) {// overflow-conscious codeint oldCapacity = elementData.length;// 计算新的容量为当前容量的1.5倍int newCapacity = oldCapacity + (oldCapacity >> 1);// 如果新容量仍然小于最小需要容量,则将新容量设置为最小需要容量if (newCapacity - minCapacity < 0)newCapacity = minCapacity;// 如果新容量超过了ArrayList允许的最大容量,则调用hugeCapacity方法确定新容量if (newCapacity - MAX_ARRAY_SIZE > 0)newCapacity = hugeCapacity(minCapacity);// 复制原数组内容到新数组elementData = Arrays.copyOf(elementData, newCapacity);
}private static int hugeCapacity(int minCapacity) {if (minCapacity < 0) // 如果minCapacity为负数,说明发生溢出,抛出异常throw new OutOfMemoryError();// 如果minCapacity大于数组最大容量,则返回Integer.MAX_VALUE,否则返回MAX_ARRAY_SIZEreturn (minCapacity > MAX_ARRAY_SIZE) ?Integer.MAX_VALUE :MAX_ARRAY_SIZE;
}private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

关键点解释:

  • 容量计算:通过 oldCapacity + (oldCapacity >> 1) 计算新容量,这是当前容量的1.5倍,意味着每次扩容后容量会增加到原来的150%。

  • 最小容量要求:如果计算出的新容量小于所需的最小容量 minCapacity,则将新容量设置为 minCapacity

  • 最大容量限制:如果新容量超过了 MAX_ARRAY_SIZEInteger.MAX_VALUE - 8),则需要调用 hugeCapacity 方法来限制新容量不超过 JVM 允许的最大值。

  • 数组复制:使用 Arrays.copyOf 方法将旧数组的内容复制到具有新容量的数组中。

异常处理:

  • 溢出处理:如果 minCapacity 计算结果为负数,表示发生了整数溢出,此时抛出 OutOfMemoryError

ArrayList 的扩容机制设计得非常高效和安全。它通过1.5倍的扩容策略减少了扩容的频率,同时通过检查最大容量限制确保了程序的稳定性。

这篇关于Java中 ArrayList 的扩容机制深度解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1103597

相关文章

Java 正则表达式URL 匹配与源码全解析

《Java正则表达式URL匹配与源码全解析》在Web应用开发中,我们经常需要对URL进行格式验证,今天我们结合Java的Pattern和Matcher类,深入理解正则表达式在实际应用中... 目录1.正则表达式分解:2. 添加域名匹配 (2)3. 添加路径和查询参数匹配 (3) 4. 最终优化版本5.设计思

Java使用ANTLR4对Lua脚本语法校验详解

《Java使用ANTLR4对Lua脚本语法校验详解》ANTLR是一个强大的解析器生成器,用于读取、处理、执行或翻译结构化文本或二进制文件,下面就跟随小编一起看看Java如何使用ANTLR4对Lua脚本... 目录什么是ANTLR?第一个例子ANTLR4 的工作流程Lua脚本语法校验准备一个Lua Gramm

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

Java Optional的使用技巧与最佳实践

《JavaOptional的使用技巧与最佳实践》在Java中,Optional是用于优雅处理null的容器类,其核心目标是显式提醒开发者处理空值场景,避免NullPointerExce... 目录一、Optional 的核心用途二、使用技巧与最佳实践三、常见误区与反模式四、替代方案与扩展五、总结在 Java

基于Java实现回调监听工具类

《基于Java实现回调监听工具类》这篇文章主要为大家详细介绍了如何基于Java实现一个回调监听工具类,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录监听接口类 Listenable实际用法打印结果首先,会用到 函数式接口 Consumer, 通过这个可以解耦回调方法,下面先写一个

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

Java字符串处理全解析(String、StringBuilder与StringBuffer)

《Java字符串处理全解析(String、StringBuilder与StringBuffer)》:本文主要介绍Java字符串处理全解析(String、StringBuilder与StringBu... 目录Java字符串处理全解析:String、StringBuilder与StringBuffer一、St

springboot整合阿里云百炼DeepSeek实现sse流式打印的操作方法

《springboot整合阿里云百炼DeepSeek实现sse流式打印的操作方法》:本文主要介绍springboot整合阿里云百炼DeepSeek实现sse流式打印,本文给大家介绍的非常详细,对大... 目录1.开通阿里云百炼,获取到key2.新建SpringBoot项目3.工具类4.启动类5.测试类6.测

Spring Boot循环依赖原理、解决方案与最佳实践(全解析)

《SpringBoot循环依赖原理、解决方案与最佳实践(全解析)》循环依赖指两个或多个Bean相互直接或间接引用,形成闭环依赖关系,:本文主要介绍SpringBoot循环依赖原理、解决方案与最... 目录一、循环依赖的本质与危害1.1 什么是循环依赖?1.2 核心危害二、Spring的三级缓存机制2.1 三

C#中async await异步关键字用法和异步的底层原理全解析

《C#中asyncawait异步关键字用法和异步的底层原理全解析》:本文主要介绍C#中asyncawait异步关键字用法和异步的底层原理全解析,本文给大家介绍的非常详细,对大家的学习或工作具有一... 目录C#异步编程一、异步编程基础二、异步方法的工作原理三、代码示例四、编译后的底层实现五、总结C#异步编程