262. Java 集合 - Java 中 ArrayList 与 LinkedList 读取元素性能大对决

文章目录

  • [262. Java 集合 - Java 中 ArrayList 与 LinkedList 读取元素性能大对决](#262. Java 集合 - Java 中 ArrayList 与 LinkedList 读取元素性能大对决)
      • [🧪 Benchmark 场景简介](#🧪 Benchmark 场景简介)
      • [⚙️ 测试代码简化展示](#⚙️ 测试代码简化展示)
      • [📊 基准测试结果总结](#📊 基准测试结果总结)
        • [✅ 读取第一个元素](#✅ 读取第一个元素)
        • [✅ 读取最后一个元素](#✅ 读取最后一个元素)
        • [⚠️ 读取中间元素](#⚠️ 读取中间元素)
      • [🧠 为什么 `LinkedList` 慢?](#🧠 为什么 LinkedList 慢?)
        • [🔍 内部结构回顾:](#🔍 内部结构回顾:)
      • [💣 Pointer Chasing & Cache Miss 示例](#💣 Pointer Chasing & Cache Miss 示例)
      • [📌 总结:读取操作谁更强?](#📌 总结:读取操作谁更强?)
      • [✅ 小贴士](#✅ 小贴士)

262. Java 集合 - Java 中 ArrayList 与 LinkedList 读取元素性能大对决

在 Java 中,我们经常面对一个问题:

🔍 哪个更适合:ArrayList 还是 LinkedList

这节我们以"读取元素"为例,用基准测试(benchmark)对两者进行深入比较。


🧪 Benchmark 场景简介

我们将对以下三种读取操作进行性能测试(使用 JMH 工具):

  1. 读取第一个元素list.get(0)
  2. 读取最后一个元素list.get(list.size() - 1)
  3. 读取中间的元素list.get(list.size() / 2)

我们对 ArrayListLinkedList 分别执行这些操作,并使用不同的列表大小进行测试。


⚙️ 测试代码简化展示

java 复制代码
List<Integer> ints = ...; // 已填充好数据
int last = ints.size() - 1;
int middle = ints.size() / 2;

ints.get(0);       // 读取第一个元素
ints.get(last);    // 读取最后一个元素
ints.get(middle);  // 读取中间元素

📊 基准测试结果总结

✅ 读取第一个元素
List 类型 Size=10 Size=100 Size=1,000 Size=10,000
ArrayList 1.18 ns 1.20 ns 1.17 ns 1.17 ns
LinkedList 1.12 ns 1.11 ns 1.12 ns 1.11 ns

👉 两者性能几乎相同。读取第一个元素是 O(1) 操作,和数据规模无关。


✅ 读取最后一个元素
List 类型 Size=10 Size=100 Size=1,000 Size=10,000
ArrayList 1.25 ns 1.23 ns 1.24 ns 1.25 ns
LinkedList 1.49 ns 1.46 ns 1.47 ns 1.48 ns

🔎 LinkedList 多了一点开销(虽然很小),但仍是 O(1),因为它有一个指向末尾的引用。


⚠️ 读取中间元素
List 类型 Size=10 Size=100 Size=1,000 Size=10,000
ArrayList 1.57 ns 1.61 ns 1.54 ns 1.53 ns
LinkedList 3.21 ns 31.11 ns 566.07 ns 7,836.10 ns

🔥 注意 LinkedList 的性能急剧下降 !这说明读取中间元素是 O(n) 操作 ------ 因为它需要顺着指针走一遍。


🧠 为什么 LinkedList 慢?

🔍 内部结构回顾:
java 复制代码
Node {
   E element;
   Node next;
   Node previous;
}
  • LinkedList双向链表
  • LinkedList 类保存两个指针:firstlast
  • 读取中间元素时必须逐个节点遍历,即"指针追踪(pointer chasing)"。

💣 Pointer Chasing & Cache Miss 示例

我们用以下代码构造更"稀疏"的链表,以人为制造 cache miss 的情况:

java 复制代码
LinkedList<Integer> ints = new LinkedList<>();
LinkedList<Integer> intsOther = new LinkedList<>();

for (int i = 0; i < LIST_SIZE; i++) {
    ints.add(i);
    for (int k = 0; k < SPARSE_INDEX; k++) {
        intsOther.add(k); // 插入无用节点,制造内存间隔
    }
}
SPARSE_INDEX 读取中间元素耗时
0 561 ns
1 602 ns
10 944 ns
100 1509 ns

➡️ 节点之间不连续,会大大降低性能。


📌 总结:读取操作谁更强?

场景 推荐使用
读取头或尾 都可以
读取中间元素 ArrayList
遍历整张表 ArrayList 更快(使用索引或增强 for 循环)

✅ 小贴士

  • 🚀 使用 JMH 进行性能测试,避免误导的微基准。
  • 📦 ArrayList 底层是数组,支持随机访问,性能稳定。
  • 🔄 LinkedList 更适合频繁插入/删除操作(尤其在头部)。
相关推荐
二哈赛车手2 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
为何创造硅基生物3 小时前
C语言 结构体内存对齐规则(通俗易懂版)
c语言·开发语言
吃好睡好便好3 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
栗子~~3 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
星寂樱易李3 小时前
iperf3 + Python-- 网络带宽、网速、网络稳定性
开发语言·网络·python
YDS8293 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
仰泳之鹅3 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
之歆4 小时前
DAY_12JavaScript DOM 完全指南(二):实战与性能篇
开发语言·前端·javascript·ecmascript
未若君雅裁4 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
cen__y5 小时前
Linux12(Git01)
linux·运维·服务器·c语言·开发语言·git