264. Java 集合 - 插入元素性能对比:LinkedList vs ArrayList

文章目录

  • [264. Java 集合 - 插入元素性能对比:LinkedList vs ArrayList](#264. Java 集合 - 插入元素性能对比:LinkedList vs ArrayList)
      • [🎯 基本原理回顾](#🎯 基本原理回顾)
        • [🔗 `LinkedList` 的插入机制:](#🔗 LinkedList 的插入机制:)
        • [📦 `ArrayList` 的插入机制:](#📦 ArrayList 的插入机制:)
    • [📊 插入操作性能对比分析](#📊 插入操作性能对比分析)
      • [📍 插入位置对性能的影响(`LinkedList`)](#📍 插入位置对性能的影响(LinkedList))
        • [🟢 插入开头(头插)表现稳定:](#🟢 插入开头(头插)表现稳定:)
        • [🔴 中间插入代价高(随着列表增长成倍增加):](#🔴 中间插入代价高(随着列表增长成倍增加):)
        • [🟡 尾部插入也很高效:](#🟡 尾部插入也很高效:)
      • [📍 插入位置对性能的影响(`ArrayList`)](#📍 插入位置对性能的影响(ArrayList))
        • [🟢 尾部添加效率极高(尤其预留容量时):](#🟢 尾部添加效率极高(尤其预留容量时):)
        • [🔶 中间插入需移动元素,代价略高:](#🔶 中间插入需移动元素,代价略高:)
        • [🔴 头部插入最慢:](#🔴 头部插入最慢:)
      • [🧨 数组扩容的隐藏成本(`ArrayList`)](#🧨 数组扩容的隐藏成本(ArrayList))
    • [🧠 总结:该用谁?在哪用?](#🧠 总结:该用谁?在哪用?)

264. Java 集合 - 插入元素性能对比:LinkedList vs ArrayList

Java 中,我们常用 LinkedListArrayList 来表示列表结构,它们在插入操作 上的性能差异尤其值得关注。本节我们通过基准测试、原理分析和实例演示,深入剖析它们在不同位置插入元素的效率表现。


🎯 基本原理回顾

🔗 LinkedList 的插入机制:
  • 每个节点包含指向前一个和后一个节点的引用。
  • 插入新节点时,只需修改相邻节点的引用指向即可。
  • 但是 :在插入中间位置时,你必须先从头或尾遍历到目标位置,这个过程非常耗时,特别是在节点不连续存储(指针跳跃+缓存未命中)。
📦 ArrayList 的插入机制:
  • 底层是一个数组。
  • 插入元素前,需要将插入点之后的所有元素整体右移一位。
  • 如果数组已满,还要扩容:重新分配一个更大的数组并复制所有元素。

📊 插入操作性能对比分析

📍 插入位置对性能的影响(LinkedList

我们测量了在开头中间末尾插入元素时的性能差异。

🟢 插入开头(头插)表现稳定:
java 复制代码
List<Integer> list = new LinkedList<>();
list.addFirst(42); // 等价于 list.add(0, 42);
SIZE Score (ns/op)
10 7.00
100 7.12
1,000 7.56

✅ 头部插入性能基本恒定,适合大量前置添加的场景。

🔴 中间插入代价高(随着列表增长成倍增加):
java 复制代码
list.add(list.size() / 2, 42);
SIZE Score (ns/op)
10 10.6
100 49.1
1,000 584.8
10,000 46,157

⚠️ 插入中间性能是灾难性的 ------ 时间复杂度约 O(n),不建议这么用!

🟡 尾部插入也很高效:
java 复制代码
list.add(42); // 或 list.addLast(42);
SIZE Score (ns/op)
10 9.13
1,000 9.79

📍 插入位置对性能的影响(ArrayList

🟢 尾部添加效率极高(尤其预留容量时):
java 复制代码
List<Integer> list = new ArrayList<>();
list.add(42);
SIZE Score (ns/op)
10 2.21
1,000 5.60

✅ 最推荐的方式,尤其是列表动态增长的主场景。

🔶 中间插入需移动元素,代价略高:
java 复制代码
list.add(list.size() / 2, 42);
SIZE Score (ns/op)
10 23.7
1,000 56.0
🔴 头部插入最慢:
java 复制代码
list.add(0, 42);
SIZE Score (ns/op)
10 22.3
10,000 717.8

⚠️ 越靠前插入,复制操作越多,性能越差。


🧨 数组扩容的隐藏成本(ArrayList

当数组容量不足时,ArrayList 会:

  1. 创建一个容量更大的新数组(通常为原大小 * 1.5)。
  2. 将旧数组内容复制过去。
  3. 插入新元素。
java 复制代码
List<Integer> list = new ArrayList<>(10);
for (int i = 0; i < 10; i++) list.add(i); // 已满
list.add(42); // 触发扩容!
SIZE 插入尾部(满) 插入中间(满) 插入头部(满)
10 19.3 ns/op 39.3 ns/op 28.2 ns/op
1,000 432.3 ns/op 451.4 ns/op 452.1 ns/op
10,000 4,140.6 ns/op 4,638.6 ns/op 4,762.7 ns/op

⚠️ 插入时触发扩容,会显著拖慢操作速度。


🧠 总结:该用谁?在哪用?

场景 推荐集合 理由
高频插入尾部 ArrayList 内存连续、写入快
高频插入头部或尾部 LinkedList 头尾插入性能稳定,特别是 addFirst/addLast
高频插入中间,数据量大 ❌ 避免使用 不管是哪种 List,中间插入都较慢(尤其 LinkedList
可预测元素数量 ArrayList 可预设容量,避免扩容成本
需要频繁插入和删除操作 LinkedList 只要不是基于 index 的随机访问
相关推荐
忧郁的Mr.Li8 小时前
SpringBoot中实现多数据源配置
java·spring boot·后端
玄同7658 小时前
从 0 到 1:用 Python 开发 MCP 工具,让 AI 智能体拥有 “超能力”
开发语言·人工智能·python·agent·ai编程·mcp·trae
czy87874758 小时前
深入了解 C++ 中的 `std::bind` 函数
开发语言·c++
消失的旧时光-19438 小时前
从 Kotlin 到 Dart:为什么 sealed 是处理「多种返回结果」的最佳方式?
android·开发语言·flutter·架构·kotlin·sealed
yq1982043011568 小时前
静思书屋:基于Java Web技术栈构建高性能图书信息平台实践
java·开发语言·前端
一个public的class8 小时前
你在浏览器输入一个网址,到底发生了什么?
java·开发语言·javascript
有位神秘人8 小时前
kotlin与Java中的单例模式总结
java·单例模式·kotlin
Jinkxs8 小时前
Gradle - 与Groovy/Kotlin DSL对比 构建脚本语言选择指南
android·开发语言·kotlin
&有梦想的咸鱼&8 小时前
Kotlin委托机制的底层实现深度解析(74)
android·开发语言·kotlin
golang学习记8 小时前
IntelliJ IDEA 2025.3 重磅发布:K2 模式全面接管 Kotlin —— 告别 K1,性能飙升 40%!
java·kotlin·intellij-idea