ArrayList 扩容倍数选 1.5 而非 2,主要与内存效率和性能优化有关。从内存对齐和碎片利用率角度看,原因如下:1. 内存对齐与分配效率Java 的 ArrayList 在扩容时会通过 Arrays.copyOf() 方法进行数组复制,而该方法在底层依赖于 JVM 的内存分配机制。如果扩容倍数为 2(即每次扩容翻倍),虽然可以减少扩容次数,但会导致大量未使用的内存空间被预分配,这可能造成内
时间:2026-05-15 21:51:31 184浏览 收藏
ArrayList 采用1.5倍而非2倍扩容,核心在于长期运行中对内存碎片率与GC压力的精细权衡:2倍扩容虽减少扩容次数,却导致数组半空闲置(如100→200后仅存120元素即浪费80位),堆积大量难以回收的“半满大数组”,加剧老年代碎片化和OOM风险;而1.5倍扩容(100→150→225)使内存利用率稳定在70%~85%,旧数组更易被完整回收,显著降低连续内存碎片和GC开销——这并非源于内存对齐或位运算玄学,而是JVM堆管理下兼顾CPU复制成本与内存空间效率的务实折中,尤其在小步增长、长期服役的业务场景中,悄然守护着系统稳定与资源韧性。

1.5 倍不是为了内存对齐,而是为了压低连续内存的碎片率——2 倍扩容在 ArrayList 场景下会快速制造大量闲置空间,尤其在长期运行或内存受限环境中容易引发 GC 压力和 OOM。
为什么 2 倍扩容会让 ArrayList 更容易产生内存碎片
ArrayList 底层是 Object[],必须申请连续堆内存。2 倍扩容(如 10 → 20 → 40 → 80)导致每次扩容后,已存元素只占新数组约 50%:
- 容量从 100 扩到 200,但只存了 120 个元素 → 80 个位置空置
- 这些空置空间无法被其他对象复用,又不能被 GC 单独回收(整个数组对象还活着)
- 多次扩容后,堆中会堆积大量“半满”的大数组对象,加剧老年代碎片化
而 1.5 倍扩容(100 → 150 → 225)让闲置比例更平缓: - 100 扩到 150 后存 120 个元素 → 仅浪费 30 个位置 - 内存利用率始终维持在 70%~85% 区间,GC 回收旧数组时更容易识别为完整可回收块
1.5 倍和内存对齐其实无关,别被位运算误导
有人看到 oldCapacity + (oldCapacity >> 1) 就联想到“对齐”,这是误解:
- 该位运算是为了高效算出 ×1.5(向下取整),不是为了地址对齐
- Java 数组本身不强制按 2 的幂次方分配,JVM 堆分配器关注的是页对齐(通常 4KB/8KB),跟数组长度是否为 2 的幂无关
- Arrays.copyOf() 分配的新数组,其起始地址由 JVM 决定,和容量数值无直接关系
- 真正影响缓存局部性的,是数组内容是否连续、访问模式是否顺序——这点 1.5 倍和 2 倍没差别
什么时候 1.5 倍会被绕过?别指望它永远生效
扩容逻辑实际是“保底策略”,不是硬约束:
- 首次添加元素时,空 ArrayList 直接扩到 10,不走 1.5 倍计算
- addAll() 一次性加 500 个元素,当前容量是 300 → grow() 会跳过 1.5 倍,直接设新容量为 500
- 若 oldCapacity 接近 Integer.MAX_VALUE,oldCapacity + (oldCapacity >> 1) 可能溢出,此时退回到最大安全值
也就是说,1.5 倍只在“小步慢增”场景下稳定起效;一旦批量操作或初始预估失误,它就自动让位给实际需求。
真正容易被忽略的点是:扩容倍数的影响不在单次操作,而在长期累积——频繁的小扩容复制带来 CPU 开销,而一次性的大扩容则吃掉大量连续堆空间。选 1.5 倍,本质是在这两者之间卡住一个能跑得久、还不太占地方的平衡点。
今天关于《ArrayList 扩容倍数选 1.5 而非 2,主要与内存效率和性能优化有关。从内存对齐和碎片利用率角度看,原因如下:1. 内存对齐与分配效率Java 的 ArrayList 在扩容时会通过 Arrays.copyOf() 方法进行数组复制,而该方法在底层依赖于 JVM 的内存分配机制。如果扩容倍数为 2(即每次扩容翻倍),虽然可以减少扩容次数,但会导致大量未使用的内存空间被预分配,这可能造成内存浪费,尤其是在对象数量增长不规律的情况下。使用 1.5 倍扩容策略,可以在保证足够容量的同时,避免过度分配内存,从而减少内存碎片的产生。2. 内存碎片利用率内存碎片指的是内存中无法被有效利用的小块空闲区域。频繁的扩容操作可能导致内存碎片化,影响后续内存分配的效率。1.5 倍扩容相比 2 倍扩容,能更灵活地适应数据量的变化,减少因“过大”扩容导致的碎片问题,提高整体内存利用率。3. 性能与资源平衡1.5 倍扩容策略是一种折中方案,在扩容频率和内存使用之间取得了较好的平衡。若扩容过快(如 2 倍),》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
295 收藏
-
479 收藏
-
422 收藏
-
341 收藏
-
243 收藏
-
357 收藏
-
478 收藏
-
184 收藏
-
494 收藏
-
429 收藏
-
279 收藏
-
183 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习