Java中ArrayList和LinkedList有什么区别举例详解

2025-02-22 05:50

本文主要是介绍Java中ArrayList和LinkedList有什么区别举例详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影...

一、底层数据结构

特性ArrayListLinkedList
实现方式基于动态数组基于双向链表
内存布局连续内存块,支持快速随机访问离散节点,每个节点包含数据及前后指针
默认初始容量10(扩容时增长 50%)无预分配容量,动态添加节点

二、核心操作性能对比

// ArrayList的随机访问示例
ArrayList<Integer> arrayList = new ArrayList<>();
arrayList.add(1);
int val1 = arrayList.get(0);  // O(1)

// LinkedList的顺序访问示例
LinkedList<Integer> linkedList = new LinkedList<>();
linkedList.add(1);
int val2 = linkedList.get(0);  // O(n)
操作ArrayList 时间复杂度LinkedList 时间复杂度
随机访问(get/set)O(1)O(n)
头部插入/删除O(n)(需移动元素)O(1)
尾部插入/删除分摊 O(1)(无扩容时 O(1))O(1)
中间插入/删除O(n)O(n)(需遍历到目标位置)

三、内存与 GC 影响

维度ArrayListLinkedList
内存占用仅存储元素 + 数组头(内存紧凑)每个节点额外存储两个指针(对象头 + 前后引用)
GC 压力整体回收高效(单个数组对象)频繁增删产生大量小对象,增加 GC 负担
缓存局部性高(连续内存,CPU 预加载命中率高)低(节点分散,缓存未命中率高)

四、扩容机制

  • ArrayList 扩容流程

    // 扩容核心逻辑(JDK17)
    int newCapacity = oldCapacity + (oldCapacity >> 1); // 1.5倍扩容
    elementData = Arrays.copyOf(elementData, newCapacity);
    
    • 代价:数据复制导致 O(n) 时间复杂度
    • 优化建议:初始化时预估容量(new ArrayList<>(initialCapacity)
  • LinkedList 无扩容:动态添加节点,但每个节点额外占用 24 字节(64 位 JVM)

五、线程安全与并发方案

方案ArrayListLinkedLisjst
默认线程安全
同步包装类Collections.synchronizedList()Collections.synchronizedList()
高并发替代方案CopyOnWr编程iteArrayListConcurrentLinkedDeque

六、工程实践场景

1. 电商商品列表展示

  • 选择 ArrayList

    • 高频读取商品信息(随机访问)
    • 批量更新时通过尾插法优化(addAll()
    List<Product> products = new ArraChina编程yList<>(10000); // 预分配容量
    

2. 实时消息队列

  • 选择 LinkedList

    • 高频头尾操作(addFirst()/removeLast()
    • 使用迭代器避免随机访问:
    LinkedList<Message> queue = new LinkedList<>();
    // 生产者
    queue.offer(new Message());
    // 消费者(高效遍历)
    Iterator<Message> it = queue.iterator();
    while(it.hasNext()) process(it.next());
    

3. 多线程日志处理器

  • 选择 CopyOnWriteArrayList

    • 写操作极少(日志初始化配置)
    • 高频遍历读取日志规则
    CopyOnWriteArrayList<LogRule&gandroidt; rules = new CopyOnWriteArrayList<>();
    // 写操作(仅在配置更新时触发)
    rules.add(new LogRule());
    // 读操作(无锁并发)
    rules.forEach(LogService::applyRule);
    

七、性能对比测试数据

测试环境:JDK17,10 万次操作,i7-11800H

测试场景ArrayList 耗时LinkedList 耗时差异原因
随机访问 1 万次2ms650ms数组 O(1) ZIgiLzTeNvs 链表 O(n) 遍历
尾部插入 1 万次3ms5ms均摊 O(1),链表节点创建开销略高
头部插入 1 万次420ms8ms数组需移动元素,链表直接修改指针
遍历所有元素15ms18ms数组缓存命中率高

八、高级特性对比

特性ArrayListLinkedList
实现接口ListList + Deque
序列化效率高(连续数据,可批量写入)低(需逐个节点处理)
内存池兼容性适合对象池化(数组整体复用)节点分散,池化效果差
批量操作优化System.arraycopy() 高效需要逐个节点操作

九、选型决策树

Java中ArrayList和LinkedList有什么区别举例详解

通过以上对比,开发者可根据具体场景选择最合适的实现:

  • 优先 ArrayList:适用于 90% 的常规场景(读多写少、内存敏感)
  • 慎用 LinkedList:仅在需要频繁头尾操作或实现双端队列时选用
  • 线程安全场景:根据写频率选择 CopyOnWriteArrayList 或同步包装类

结论

  • ArrayList 更适合:如果你的操作主要集中在随机访问和列表末尾的插入删除,且你希望节省内存,ArrayList 是首选。

  • LinkedList 更适合:如果你的操作主要是频繁在列表中间进行插入和删除,而随机访问较少,LinkedList 会表现得更好。

总结 

到此这篇关于Java中ArrayList和LinkedList有什么区别的文章就介绍到这了,更多相关Java中ArrayList和LinkedList区别内容请搜索编程China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持China编程(www.chinasem.cn)!

这篇关于Java中ArrayList和LinkedList有什么区别举例详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

Redis 的 SUBSCRIBE命令详解

《Redis的SUBSCRIBE命令详解》Redis的SUBSCRIBE命令用于订阅一个或多个频道,以便接收发送到这些频道的消息,本文给大家介绍Redis的SUBSCRIBE命令,感兴趣的朋友跟随... 目录基本语法工作原理示例消息格式相关命令python 示例Redis 的 SUBSCRIBE 命令用于订

Vue和React受控组件的区别小结

《Vue和React受控组件的区别小结》本文主要介绍了Vue和React受控组件的区别小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录背景React 的实现vue3 的实现写法一:直接修改事件参数写法二:通过ref引用 DOMVu

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java