详解:LinkedList的工作原理和实现

LinkedList 是 Java 中基于双向链表实现的线性表结构,支持高效的插入和删除操作,但随机访问性能较低。本文将从核心原理和实现细节展开分析:

一、核心结构

1. 节点(Node)

每个节点包含前驱指针后继指针数据,形成链式结构:

swift 复制代码
java
private static class Node<E> {
    E item;         // 数据
    Node<E> next;   // 后继节点
    Node<E> prev;   // 前驱节点
}

2. 链表头尾

  • 头指针(first):指向链表第一个节点。
  • 尾指针(last):指向链表最后一个节点。
  • 大小(size):记录链表当前元素数量。

二、核心操作实现

1. 插入操作

  • 头部插入

    创建新节点,将新节点的next指向原头节点,更新first指针:

    ini 复制代码
    java
    void linkFirst(E e) {
        Node<E> f = first;
        Node<E> newNode = new Node<>(null, e, f);
        first = newNode;
        if (f == null) last = newNode; // 空链表时更新尾指针
        else f.prev = newNode;
        size++;
    }
  • 尾部插入

    创建新节点,将原尾节点的next指向新节点,更新last指针:

    ini 复制代码
    java
    void linkLast(E e) {
        Node<E> l = last;
        Node<E> newNode = new Node<>(l, e, null);
        last = newNode;
        if (l == null) first = newNode; // 空链表时更新头指针
        else l.next = newNode;
        size++;
    }
  • 中间插入

    在指定位置找到前驱和后继节点,调整前后指针:

    ini 复制代码
    java
    void linkBefore(E e, Node<E> succ) {
        Node<E> pred = succ.prev;
        Node<E> newNode = new Node<>(pred, e, succ);
        succ.prev = newNode;
        if (pred == null) first = newNode; // 插入到头部
        else pred.next = newNode;
        size++;
    }

2. 删除操作

  • 删除头节点

    断开原头节点的next指针,更新first

    ini 复制代码
    java
    E unlinkFirst(Node<E> f) {
        E element = f.item;
        Node<E> next = f.next;
        f.item = null;
        f.next = null; // 帮助GC
        first = next;
        if (next == null) last = null; // 链表为空时更新尾指针
        else next.prev = null;
        size--;
        return element;
    }
  • 删除尾节点

    类似头部删除,更新last指针。

  • 删除中间节点

    调整前驱和后继节点的指针:

    ini 复制代码
    java
    E unlink(Node<E> x) {
        Node<E> pred = x.prev;
        Node<E> next = x.next;
        if (pred == null) first = next; // 删除的是头节点
        else pred.next = next;
        if (next == null) last = pred;  // 删除的是尾节点
        else next.prev = pred;
        x.item = null; // 清理数据
        size--;
        return x.item;
    }

3. 查找操作

  • 按索引查找

    根据索引位置选择从头或尾遍历,时间复杂度 O(n)

    ini 复制代码
    java
    Node<E> node(int index) {
        if (index < (size >> 1)) { // 前半部分从头遍历
            Node<E> x = first;
            for (int i = 0; i < index; i++)
                x = x.next;
            return x;
        } else { // 后半部分从尾遍历
            Node<E> x = last;
            for (int i = size - 1; i > index; i--)
                x = x.prev;
            return x;
        }
    }

三、性能分析

操作 时间复杂度 说明
插入/删除头尾 O(1) 直接调整头尾指针
插入/删除中间 O(n) 需遍历到目标位置
随机访问(get) O(n) 必须从头或尾遍历链表
迭代遍历 O(n) 顺序访问所有节点

四、线程安全性

  • 非线程安全:多线程并发修改会导致数据不一致。

  • 替代方案

    • 使用 Collections.synchronizedList(new LinkedList<>())
    • 使用并发容器如 ConcurrentLinkedQueue

五、与 ArrayList 对比

特性 LinkedList ArrayList
底层结构 双向链表 动态数组
内存占用 更高(存储指针) 更低(连续内存)
插入/删除性能 O(1)(头尾) O(n)(需移动元素)
随机访问性能 O(n) O(1)
扩容机制 无需扩容 自动扩容(1.5倍)

六、应用场景

  1. 频繁插入/删除 :如实现队列(Queue)或栈(Deque)。
  2. 动态数据管理:数据规模不确定且需要高效增删。
  3. 实现高级数据结构:LRU缓存、任务调度队列等。

七、实现细节优化

  1. 双向遍历优化

    查找时根据索引位置选择从头或尾遍历(见 node(int index) 方法)。

  2. 空值支持

    允许存储 null 值,节点通过 item == null 判断。

  3. 迭代器设计

    提供 ListIterator 支持双向遍历和修改操作:

    scss 复制代码
    java
    public ListIterator<E> listIterator(int index) {
        checkPositionIndex(index);
        return new ListItr(index);
    }
  4. Fail-Fast 机制

    迭代过程中检测结构性修改(通过 modCount 计数器),防止并发修改异常。

八、注意事项

  1. 避免随机访问
    不要用 get(index) 遍历链表,应使用迭代器或增强 for 循环。
  2. 内存开销
    每个元素需额外存储两个指针,内存占用高于 ArrayList
  3. 对象生命周期
    删除节点后及时清理 item 和指针(置为 null),避免内存泄漏。

通过双向链表的灵活指针操作,LinkedList 在特定场景下表现出色,但需根据需求权衡其优缺点。理解其实现机制有助于在开发中合理选择数据结构。

相关推荐
阿拉斯攀登3 分钟前
【RK3576 安卓 JNI/NDK 系列 04】JNI 核心语法(下):字符串、数组与对象操作
android·驱动开发·rk3568·瑞芯微·rk安卓驱动·jni字符串操作
GoodStudyAndDayDayUp4 分钟前
RUO-VUE-PRO权限关联sql
java·数据库·sql
2501_915909065 分钟前
不用越狱就看不到 iOS App 内部文件?使用 Keymob 查看和导出应用数据目录
android·ios·小程序·https·uni-app·iphone·webview
llxxyy卢9 分钟前
web部分中等题目
android·前端
轩情吖12 分钟前
MySQL之事务管理
android·后端·mysql·adb·事务·隔离性·原子性
⑩-19 分钟前
RabbitMQ 架构和工作原理?RabbitMQ 延迟队列如何实现?
java·分布式·架构·rabbitmq
子非鱼@Itfuture21 分钟前
try-catch和try-with-resources区别是什么?try{}catch(){}和try(){}catch(){}有什么好处?
java·开发语言
Morwit35 分钟前
*【力扣hot100】 215. 数组中的第K个最大元素
数据结构·c++·算法·leetcode·职场和发展
ab15151736 分钟前
3.20二刷基础121、127,完成进阶61、62
数据结构·算法·排序算法
I_LPL37 分钟前
day58 代码随想录算法训练营 图论专题11
数据结构·算法·图论