LinkedList数据结构链表

LinkedList在Java中是一个实现了ListDeque接口的双向链表。它允许我们在列表的两端添加或删除元素,同时也支持在列表中间插入或移除元素。在分析LinkedList之前,需要理解链表这种数据结构:

  • 链表:链表是一种动态数据结构,由一系列节点组成,每个节点包含数据部分和指向列表中下一个节点的引用。
  • 双向链表:每个节点都有两个链接,一个指向前一个节点,另一个指向后一个节点。

LinkedList 的数据结构

LinkedList中,每个元素都是一个节点,每个节点包含三个部分:存储的数据项、指向前一个节点的引用和指向后一个节点的引用。

java 复制代码
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;
    }
}

LinkedList 的核心方法

LinkedList实现了List接口,所以它具有诸如add, remove, get, set等方法,同时也实现了Deque接口,这意味着它也具有offer, poll, push, pop等方法。

源码解析(简化版)

以下是LinkedList中一些关键方法的简化版源码:

java 复制代码
public class LinkedList<E>
    extends AbstractSequentialList<E>
    implements List<E>, Deque<E>, Cloneable, java.io.Serializable {

    transient int size = 0;
    transient Node<E> first;
    transient Node<E> last;

    public LinkedList() {
    }

    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++;
    }

    public E remove() {
        return unlinkFirst(first);
    }

    private E unlinkFirst(Node<E> f) {
        final E element = f.item;
        final Node<E> next = f.next;
        f.item = null;
        f.next = null; 
        first = next;
        if (next == null)
            last = null;
        else
            next.prev = null;
        size--;
        modCount++;
        return element;
    }

    public E get(int index) {
        checkElementIndex(index);
        return node(index).item;
    }

    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;
        }
    }

    // 省略其他细节实现
}

代码演示

下面是LinkedList的一个简单使用示例:

java 复制代码
import java.util.LinkedList;

public class Main {
    public static void main(String[] args) {
        LinkedList<String> friends = new LinkedList<>();

        // 添加元素
        friends.add("Alice");
        friends.add("Bob");
        friends.add("Charlie");

        // 打印所有元素
        System.out.println("Initial LinkedList: " + friends);

        // 删除第一个元素
        friends.remove();

        // 打印删除元素后的列表
        System.out.println("After removal: " + friends);

        // 获取特定位置的元素
        String friend = friends.get(1);
        System.out.println("Second friend: " + friend);
    }
}

细节分析

使用LinkedList时,有几个细节需要注意:

  • 动态扩展 :由于LinkedList是基于节点的,因此它可以动态地添加或删除节点而不需要像数组那样重新分配整个数据结构。
  • 随机访问效率低 :获取特定索引的元素时,LinkedList必须从头开始(或从尾开始,取决于哪边更近)遍历节点。因此,随机访问的效率远低于数组。
  • 插入和删除效率高 :在任何位置插入或删除元素时,LinkedList只需要改变几个引用,这使得插入和删除操作非常快速。
  • 内存开销 :与数组相比,LinkedList的每个元素都需要额外的内存空间来存储前后节点的引用。

通过以上解析,我们可以深入理解LinkedList的工作原理和设计。这有助于开发者在需要适合的数据结构时作出明智的选择。对于需要频繁插入和删除操作的场景,LinkedList可能是一个不错的选择。然而,如果需要快速通过索引访问元素,那么ArrayList可能是更好的选择。

相关推荐
C++忠实粉丝3 小时前
前缀和(8)_矩阵区域和
数据结构·c++·线性代数·算法·矩阵
ZZZ_O^O3 小时前
二分查找算法——寻找旋转排序数组中的最小值&点名
数据结构·c++·学习·算法·二叉树
代码雕刻家4 小时前
数据结构-3.9.栈在递归中的应用
c语言·数据结构·算法
Kalika0-06 小时前
猴子吃桃-C语言
c语言·开发语言·数据结构·算法
代码雕刻家6 小时前
课设实验-数据结构-单链表-文教文化用品品牌
c语言·开发语言·数据结构
小字节,大梦想7 小时前
【C++】二叉搜索树
数据结构·c++
我是哈哈hh8 小时前
专题十_穷举vs暴搜vs深搜vs回溯vs剪枝_二叉树的深度优先搜索_算法专题详细总结
服务器·数据结构·c++·算法·机器学习·深度优先·剪枝
丶Darling.8 小时前
LeetCode Hot100 | Day1 | 二叉树:二叉树的直径
数据结构·c++·学习·算法·leetcode·二叉树
labuladuo5209 小时前
Codeforces Round 977 (Div. 2) C2 Adjust The Presentation (Hard Version)(思维,set)
数据结构·c++·算法
Indigo_code9 小时前
【数据结构】【链表代码】合并有序链表
数据结构·windows·链表