数据结构-LinkedList原理

1. 简介

1.1 同 ArrayList 区别是什么?

  • ArrayList
    • 优点:底层是数组,检索非常快
    • 缺点:底层是数组,所以扩容需要申请新的数组并且要把老数据拷贝到新数组,代价比较大
  • LinkedList
    • 优点:进行增删改仅需要改变两个节点的指针引用即可,较快
    • 缺点:底层是链表,通过下标寻找元素,需要从链表头开始一个节点一个节点往下寻找,检索比较慢

1.2 继承关系

相较于 ArrayListLinkedList 多实现了接口 Queue (队列)、Deque (双端队列)。

队列的含义是先进先出,双端队列顾名思义就是可以把队列的首尾都当作队列处理,也就是首尾都可以增删元素,说明 LinkedList 除了队列先进先出的能力还具有栈先进后出的能力。

2. LinkedList 方法

2.1 构造方法

构造器 描述
LinkedList() 构造一个空列表。
LinkedList​(Collection<? extends E> c) 按照集合的迭代器返回的顺序构造一个包含指定集合元素的列表。

2.2 具体方法

变量和类型 方法 方法来源 操作 位置 列表内为空处理
void add​(int index, E element) List 指定位置
boolean add​(E e) Queue 末尾
void addFirst​(E e) Deque 开头
void addLast​(E e) Deque 末尾
void clear() Collection 所有位置
Iterator<E> descendingIterator() Deque 迭代
E element() Queue 检索 开头
E get​(int index) List 检索 指定位置
E getFirst() Deque 检索 开头
E getLast() Deque 检索 末尾
boolean offer​(E e) Queue 末尾
boolean offerFirst​(E e) Deque 开头
boolean offerLast​(E e) Deque 末尾
E peek() Queue 检索 开头
E peekFirst() Deque 检索 开头
E peekLast() Deque 检索 末尾
E poll() Queue 检索,删 开头 返回 null
E pollFirst() Deque 检索,删 开头 返回 null
E pollLast() Deque 检索,删 末尾 返回 null
E pop() Deque 检索,删 开头 异常
void push​(E e) Deque 末尾
E remove() Queue 检索 开头 异常
E remove​(int index) List 检索,删 指定位置 异常
boolean remove​(Object o) List 检索,删 第一个匹配项 异常
E removeFirst() Deque 检索,删 开头 异常
boolean removeFirstOccurrence​(Object o) Deque 第一个匹配项 异常
E removeLast() Deque 检索,删 末尾 异常
boolean removeLastOccurrence​(Object o) Deque 最后一个匹配项 异常
E set​(int index, E element) List 设置 指定位置

2.3 重点方法原理讲解

add​(int index, E element)

LinkedList 中的每一个节点都有两个指针 next(指向下一个节点),prev(指向上一个节点)。 如上图,如果想要用 NEW 22 替代,需要做的工作如下:

  • 2next 指针与 prev 指针指向 null,此时 2 便不会被引用,也不会引用其他节点
  • NEW 2next 指针指向 3prev 指针指向 1
  • 1next 指针指向 NEW 2
  • 3prev 指针指向 NEW 2
  • 注意,如果替换的节点是 Head 节点/ Tail 节点,还需要改变两个指针的指向引用

remove​(int index)

如上图,如果需要删除一个节点,需要做的工作如下:

  • 2next 指针与 prev 指针指向 null,此时 2 便不会被引用,也不会引用其他节点
  • 1next 指针指向 3
  • 3prev 指针指向 1
  • 注意,如果删除的节点是 Head 节点/ Tail 节点,还需要改变两个指针的指向引用

3. 总结

对于查操作较多的场景用 ArrayList 性能更好,可以直接获取到内存中的地址,不用遍历。

对于增删操作多的场景用 LinkedList ,只需要改变指针指向即可,不需要进行扩容拷贝。

LinkedList 具有队列和栈的能力,有相关需求在使用时可以考虑这两个特性。

小贴士:可以用 LinkedList 的队列特性对树结构进行广度优先搜索

java 复制代码
class TreeNode {
    private int val;
    private TreeNode left;
    private TreeNode right;
}

public static void main(String[] args) {
    TreeNode root;

    LinkedList<TreeNode> queue = new LinkedList<>();
    queue.add(root);

    while (!queue.isEmpty()) {
        for (int i = queue.size() - 1 /*拷贝队列当前元素数量*/; i >= 0; i--) {
            TreeNode treeNode = queue.poll();
            TreeNode left = treeNode.left;
            TreeNode right = treeNode.right;

            System.out.println(treeNode.val);

            if (left != null) {
                queue.offer(left);
            }
            if (right != null) {
                queue.offer(right);
            }
        }
    }
}
相关推荐
camellias_6 分钟前
SpringBoot(二十三)SpringBoot集成JWT
java·spring boot·后端
tebukaopu1487 分钟前
springboot如何获取控制层get和Post入参
java·spring boot·后端
昔我往昔7 分钟前
SpringBoot 创建对象常见的几种方式
java·spring boot·后端
q567315238 分钟前
用 PHP或Python加密字符串,用iOS解密
java·python·ios·缓存·php·命令模式
灭掉c与java14 分钟前
第三章springboot数据访问
java·spring boot·后端
啊松同学42 分钟前
【Java】设计模式——工厂模式
java·后端·设计模式
枫叶_v1 小时前
【SpringBoot】20 同步调用、异步调用、异步回调
java·spring boot·后端
鸣弦artha1 小时前
蓝桥杯——杨辉三角
java·算法·蓝桥杯·eclipse
大波V51 小时前
设计模式-参考的雷丰阳老师直播课
java·开发语言·设计模式
计算机-秋大田2 小时前
基于微信小程序的平安驾校预约平台的设计与实现(源码+LW++远程调试+代码讲解等)
java·spring boot·微信小程序·小程序·vue·课程设计