likes
comments
collection
share

数据结构-LinkedList原理

作者站长头像
站长
· 阅读数 20

1. 简介

1.1 同 ArrayList 区别是什么?

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

1.2 继承关系

数据结构-LinkedList原理

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

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

2. LinkedList 方法

2.1 构造方法

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

2.2 具体方法

变量和类型方法方法来源操作位置列表内为空处理
voidadd​(int index, E element)List指定位置
booleanadd​(E e)Queue末尾
voidaddFirst​(E e)Deque开头
voidaddLast​(E e)Deque末尾
voidclear()Collection所有位置
Iterator<E>descendingIterator()Deque迭代
Eelement()Queue检索开头
Eget​(int index)List检索指定位置
EgetFirst()Deque检索开头
EgetLast()Deque检索末尾
booleanoffer​(E e)Queue末尾
booleanofferFirst​(E e)Deque开头
booleanofferLast​(E e)Deque末尾
Epeek()Queue检索开头
EpeekFirst()Deque检索开头
EpeekLast()Deque检索末尾
Epoll()Queue检索,删开头返回 null
EpollFirst()Deque检索,删开头返回 null
EpollLast()Deque检索,删末尾返回 null
Epop()Deque检索,删开头异常
voidpush​(E e)Deque末尾
Eremove()Queue检索开头异常
Eremove​(int index)List检索,删指定位置异常
booleanremove​(Object o)List检索,删第一个匹配项异常
EremoveFirst()Deque检索,删开头异常
booleanremoveFirstOccurrence​(Object o)Deque第一个匹配项异常
EremoveLast()Deque检索,删末尾异常
booleanremoveLastOccurrence​(Object o)Deque最后一个匹配项异常
Eset​(int index, E element)List设置指定位置

2.3 重点方法原理讲解

add​(int index, E element)

数据结构-LinkedList原理

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

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

remove​(int index)

数据结构-LinkedList原理

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

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

3. 总结

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

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

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

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

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