【Java】ArrayList和LinkedList的区别是什么

目录

1. 数据结构

2. 性能特点

3. 源码分析

4. 代码演示

5. 细节和使用场景


ArrayListLinkedList 分别代表了两类不同的数据结构:动态数组和链表。它们都实现了 Java 的 List 接口,但是有着各自独特的特点和性能表现。

1. 数据结构

  • ArrayList 是基于可调整大小的数组实现的。它允许快速随机访问,因为内部元素可通过数组索引直接访问。
  • LinkedList 是基于双向链表实现的。链表中的每个元素都包含了对其前一个和后一个元素的引用,允许双向遍历。

2. 性能特点

特性/操作ArrayListLinkedList
随机访问O(1)O(n)
添加元素(一般)O(1) (摊销时间)O(1)
在末尾添加元素O(1) (摊销时间)O(1)
在中间/开始添加元素O(n)O(1)
删除元素(一般)O(n)O(1)
内存开销较小 (只存数据)较大 (数据 + 两个引用)

3. 源码分析

  • ArrayList 源码关键部分
public class ArrayList<E> extends AbstractList<E> implements List<E>, RandomAccess, Cloneable, java.io.Serializable {transient Object[] elementData; // 存储数据private void ensureCapacityInternal(int minCapacity) {if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);}if (minCapacity - elementData.length > 0) {grow(minCapacity);}}private void grow(int minCapacity) {// 扩容逻辑}public E get(int index) {rangeCheck(index);return elementData[index];}public boolean add(E e) {ensureCapacityInternal(size + 1);  // 确保容量elementData[size++] = e;return true;}// ...省略其他方法
}

 

ArrayList的核心是一个数组。当添加元素会超过当前数组大小时,会触发一个“扩容”操作,通常是将数组大小增加到当前大小的1.5倍。

  • LinkedList 源码关键部分
public class LinkedList<E> extends AbstractSequentialList<E> implements List<E>, Deque<E>, Cloneable, java.io.Serializable {transient Node<E> first;transient Node<E> last;private static class Node<E> {E item;Node<E> next;Node<E> prev;Node(Node<E> prev, E element, Node<E> next) {this.item = element;this.next = next;this.prev = prev;}}public boolean add(E e) {linkLast(e);return true;}void linkLast(E e) {final Node<E> l = last;final Node<E> newNode = new Node<>(l, e, null);last = newNode;if (l == null)first = newNode;elsel.next = newNode;size++;modCount++;}// ...省略其他方法
}

 

LinkedList中的每个元素都是一个节点对象,包含了数据和两个指向其它节点的引用。

4. 代码演示

以下代码展示了ArrayListLinkedList的基本使用:

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;public class ListExample {public static void main(String[] args) {List<String> arrayList = new ArrayList<>();List<String> linkedList = new LinkedList<>();// 添加元素arrayList.add("Element1");linkedList.add("Element1");// 在列表中间插入元素arrayList.add(0, "Element2"); // O(n)linkedList.add(0, "Element2"); // O(1), 只需要改变引用// 获取元素String elementFromArrayList = arrayList.get(1); // O(1)String elementFromLinkedList = linkedList.get(1); // O(n), 需要从头遍历链表// 删除元素arrayList.remove(0); // O(n)linkedList.remove(0); // O(1), 只需要改变引用}
}

 

5. 细节和使用场景

  • ArrayList

    • 优先选择,当需要频繁访问列表中的元素。
    • 注意处理扩容操作,可能会导致短暂的性能下降。
    • 更低的内存占用。
  • LinkedList

    • 当需要频繁进行添加和删除操作,尤其是在列表的开头或中间时,可以考虑使用。
    • 每个元素占用更多内存,因为存储了两个额外的引用。

理解这些区别和细节可以帮助你做出适合你应用场景的数据结构选择。尽管LinkedList在某些操作中有其优势,但由于内存使用和大多数操作中的性能影响,ArrayList通常是默认首选。只有在特定的、频繁进行插入和删除的场景下,LinkedList才是更好的选择。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/459515.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CPP项目:Boost搜索引擎

1.项目背景 对于Boost库来说&#xff0c;它是没有搜索功能的&#xff0c;所以我们可以实现一个Boost搜索引擎来实现一个简单的搜索功能&#xff0c;可以更快速的实现Boost库的查找&#xff0c;在这里&#xff0c;我们实现的是站内搜索&#xff0c;而不是全网搜索。 2.对于搜索…

STM32学习笔记——定时器

目录 一、定时器功能概述 1、基本定时器&#xff08;TIM6&TIM7&#xff09; 工作原理 时序 2、通用计时器&#xff08;TIM2&TIM3&TIM4&TIM5&#xff09; 时钟源 外部时钟源模式1&2 外部时钟源模式2 外部时钟源模式1 定时器的主模式输出 输入捕获…

【开源】SpringBoot框架开发超市账单管理系统 JAVA+Vue+SpringBoot+MySQL

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块三、系统设计3.1 总体设计3.2 前端设计3.3 后端设计在这里插入图片描述 四、系统展示五、核心代码5.1 查询供应商5.2 查询商品5.3 新增超市账单5.4 编辑超市账单5.5 查询超市账单 六、免责说明 一、摘要 1.1 项目介绍 基于…

MySQL 日志管理

4.6&#xff09;日志管理 MySQL 支持丰富的日志类型&#xff0c;如下&#xff1a; 事务日志&#xff1a;transaction log 事务日志的写入类型为 "追加"&#xff0c;因此其操作为 "顺序IO"&#xff1b; 通常也被称为&#xff1a;预写式日志 write ahead…

服务器运存使用率多少正常?

服务器运存使用率多少正常&#xff0c;这是一个相对主观的问题&#xff0c;因为服务器的正常运行不仅取决于运存使用率&#xff0c;还与服务器的工作负载、应用程序的特性和需求、服务器的配置和用途等多种因素有关。然而&#xff0c;一般来说&#xff0c;大多数服务器在运存使…

WordPress如何实现随机显示一句话经典语录?怎么添加到评论框中?

我们在一些WordPress网站的顶部或侧边栏或评论框中&#xff0c;经常看到会随机显示一句经典语录&#xff0c;他们是怎么实现的呢&#xff1f; 其实&#xff0c;boke112百科前面跟大家分享的『WordPress集成一言&#xff08;Hitokoto&#xff09;API经典语句功能』一文中就提供…

Netty中解决粘包/半包

目录 什么是TCP粘包半包&#xff1f; TCP 粘包/半包发生的原因 解决粘包半包 channelRead和channelReadComplete区别 什么是TCP粘包半包&#xff1f; 假设客户端分别发送了两个数据包 D1 和 D2 给服务端&#xff0c;由于服务端一次读取到的字节数是不确定的&#xff0c;故可…

基于微信小程序的书籍阅读系统,附源码

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

Redis核心技术与实战【学习笔记】 - 27.限制Redis Cluster规模的因素(通信开销)

简述 Redis Cluster 能保存的数据量以及支撑的吞吐量&#xff0c;跟集群实例规模相关。 Redis 官方给出了 Redis Cluster 的规模上线&#xff0c;就是一个集群运行 1000 个实例。 其实&#xff0c;限定 Redis Cluster 集群规模的一个关键因素就是&#xff0c;实例间的通信开销…

使用SD-WAN进行企业网络升级的必要性

随着企业业务的不断扩大&#xff0c;企业的网络建设成为业务成功的基础。对于中小企业而言&#xff0c;提高生产力和竞争力&#xff0c;电脑化和信息化已成为不可逆转的趋势。各种关键业务应用如办公室自动化、数据库、ERP、CRM、物流供应链等的增加&#xff0c;对网络的速度、…

JAVA生产使用登录校验模式

背景 目前我们的服务在用户登录时&#xff0c;会先通过登录接口进行密码校验。一旦验证成功&#xff0c;后端会利用UUID生成一个独特的令牌&#xff08;token&#xff09;&#xff0c;并将其存储在Redis缓存中。同时&#xff0c;前端也会将该令牌保存在本地。在后续的接口请求…

基于51 单片机的交通灯系统 源码+仿真+ppt

主要内容&#xff1a; 1&#xff09;南北方向的绿灯、东西方向的红灯同时亮40秒。 2&#xff09;南北方向的绿灯灭、黄灯亮5秒&#xff0c;同时东西方向的红灯继续亮。 3&#xff09;南北方向的黄灯灭、左转绿灯亮&#xff0c;持续20秒&#xff0c;同时东西方向的红灯继续…