当前位置: 首页 > article >正文

【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;
        else
            l.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.kler.cn/a/231496.html

相关文章:

  • 身份证号码校验
  • webpack配置
  • Nginx Spring boot指定域名跨域设置
  • sqli—labs靶场 5-8关 (每日4关练习)持续更新!!!
  • 4.STM32之通信接口《精讲》之USART通信---实验串口发送程序
  • React Native 全栈开发实战班 - 数据管理与状态之Zustand应用
  • C语言的循环结构
  • 深入探索 Express.js 的高级特性
  • DevOps落地笔记-20|软件质量:决定系统成功的关键
  • 使用x86架构+Nvidia消费显卡12G显存,搭建智能终端,将大模型本地化部署,说不定是未来方向,开源交互机器人设计
  • Unity AnimationRigging无法修改权重?
  • 【代理模式】
  • 【iOS ARKit】3D人体姿态估计实例
  • SpringCloud-Eureka原理分析
  • 2.0 Zookeeper 安装配置
  • WordPress如何实现随机显示一句话经典语录?怎么添加到评论框中?
  • Maven构建OSGI+HttpServer应用
  • 【Flask + AI】接入CHATGLM API 实现翻译接口
  • UML 2.5图形库
  • C++算法学习心得八.动态规划算法(1)
  • 前端性能优化:提升网站加载速度的终极指南
  • LeetCode回溯算法的解题思路
  • 使用 Matlab 拟合函数
  • 无广告iOS获取设备UDID 简单方便快捷
  • 常用ES技巧二
  • Transformer实战-系列教程11:SwinTransformer 源码解读4(WindowAttention类)