Java面试专项——集合专题三(LinkedList)

目录

ArrayList引发的思考

LinkedList底层原理

LinkedList强大在哪里

手写LinkedList加深印象 

LinkedList集合中迭代器的fail-fast机制


ArrayList引发的思考

优点:查询快

缺点:

1、增删慢,消耗CPU性能(情况一:向指定索引上添加或删除元素,需要往前或往后移动索引后的元素  情况二:如果原数组中空间不够了,需要扩容,创建一个新的数组然后将原数组中的元素复制到新数组上)

2、比较浪费内存空间(申请的空间必须是连续的,并且并不是申请几个空间就用几个空间)

那么有没有一个集合能够弥补ArrayList的缺陷呢?这里我们就要提到双向链表了。

说起数据结构中的链表,我们不得不提这种数据结构的优缺点:

优点:添加和删除元素比较快,因为只是移动指针,并且不需要判断是否需要扩容

缺点:查询和遍历效率比较低,因为链表不能想顺序表那样随机存取,在第i个位置上执行存或者取的操作,顺序表仅需访问一次,而链表需要从头开始依次访问i次

而LinkedList就是基于双向循环链表实现的首先来看一个LinkedList的继承实现树:

Java面试专项——集合专题三(LinkedList)_第1张图片

LinkedList不仅是双向循环链表的实现,除了可以当做单链表、双链表、循环链表来操作外,它实现了Deque接口,它还可以当做栈、队列和双端队列来使用。

LinkedList底层原理

首先我们来重点说说双向链表这个数据结构

Java面试专项——集合专题三(LinkedList)_第2张图片

每一个节点都由3个部分组成,一个是数据本身item,一个是指向下一个节点的next指针,还有就是指向上一个节点的prev指针,另外,双向链表还有一个 first 指针,指向头节点,和 last 指针,指向尾节点。在LinkedList类中通过私有的静态内部类Node作为每一个数据的封装。具体实现如下: 

private static class Node {
        E item;
        Node next;
        Node prev;

        Node(Node prev, E element, Node next) {
            this.item = element;
            this.next = next;
            this.prev = prev;
        }
    }

再看该类的成员变量

Java面试专项——集合专题三(LinkedList)_第3张图片

 成员变量size用来记录双向链表的大小,first节点用来指向链表的头,last用来指向链表的尾

再看类的构造方法

public LinkedList() {
    }

    /**
     * Constructs a list containing the elements of the specified
     * collection, in the order they are returned by the collection's
     * iterator.
     *
     * @param  c the collection whose elements are to be placed into this list
     * @throws NullPointerException if the specified collection is null
     */
    public LinkedList(Collection c) {
        this();
        addAll(c);
    }

该类有两个构造方法,一个是空参数构造方法,另外一个是通过已有的集合进行构造。我们再来看看通过集合构造里面调用的addAll()方法做了什么

public boolean addAll(Collection c) {
        return addAll(size, c);
    }
public boolean addAll(int index, Collection c) {
        checkPositionIndex(index);

        Object[] a = c.toArray();
        int numNew = a.length;
        if (numNew == 0)
            return false;

        Node pred, succ;
        if (index == size) {
            succ = null;
            pred = last;
        } else {
            succ = node(index);
            pred = succ.prev;
        }

        for (Object o : a) {
            @SuppressWarnings("unchecked") E e = (E) o;
            Node newNode = new Node<>(pred, e, null);
            if (pred == null)
                first = newNode;
            else
                pred.next = newNode;
            pred = newNode;
        }

        if (succ == null) {
            last = pred;
        } else {
            pred.next = succ;
            succ.prev = pred;
        }

        size += numNew;
        modCount++;
        return true;
    }

 可见,调用集合调用方法,如果传入的集合不为空,就将集合中的元素遍历为每个值封装成一个Node节点,然后依次链接到链表上。

注意如果传入的集合为空,那么会报出空指针异常 

Java面试专项——集合专题三(LinkedList)_第4张图片

接下来看看LinkedList的基本操作:添加,删除,遍历,查询等

先看添加,从双向链表的结构来看,添加元素可以在链表的头、尾、以及中间的任意位置添加新的元素。因为 LinkedList 有头指针和尾指针,所以在表头或表尾进 行插入元素只需要 O(1) 的时间,而在指定位置插入元素则需要先遍历一下链表, 所以复杂度为 O(n)。首先看看在头部添加元素:

Java面试专项——集合专题三(LinkedList)_第5张图片

 从头部添加元素,只要把头指针first指向新的node节点,新的node节点的next指针指向原先first指针指向的node,再把原先first指针指向的node的prev指针指向新的node节点就可以了

private void linkFirst(E e) {
        final Node f = first; //使用临时node
        final Node newNode = new Node<>(null, e, f); 
        //封装新的node,并把新node的nex指向f
        first = newNode;
        if (f == null) //判断first是否为空
            last = newNode;
        else
            f.prev = newNode; //把f的prev指向新的node
        size++; //链表长度加1
        modCount++; //记录链表被修改的次数
    }

在尾部添加,其实和在头部添加一样,只是把first换成了last,逻辑一样 

void linkLast(E e) {
        final Node l = last;
        final Node newNode = new Node<>(l, e, null);
        last = newNode;
        if (l == null)
            first = newNode;
        else
            l.next = newNode;
        size++;
        modCount++;
    }

那么在中间位置添加元素呢?

Java面试专项——集合专题三(LinkedList)_第6张图片

void linkBefore(E e, Node succ) { //表示在在succ节点前面添加e元素
        // assert succ != null;
        final Node pred = succ.prev; //获取succ的前面节点
        final Node newNode = new Node<>(pred, e, succ); //把e封装成节点,并把prev指向succ前面节点,把next指向succ节点
        succ.prev = newNode; //然后把succ的prev指向新的节点
        if (pred == null)
            first = newNode;
        else
            pred.next = newNode; //把succ的前节点的next只想新的节点
        size++; //链表长度+1
        modCount++; //修改次数+1
    }
void linkLast(E e) {
        final Node l = last;
        final Node newNode = new Node<>(l, e, null);
        last = newNode;
        if (l == null)
            first = newNode;
        else
            l.next = newNode;
        size++;
        modCount++;
    }

在指定元素之前、之后添加元素,需要改动被插入以及元素前后或者后面的节点的指针

从头部、尾部、中间位置删除、修改元素都与添加类似。如果学过数据结构,那么这些操作自然很熟悉,甚至我前面说的都显得是多余的。

LinkedList强大在哪里

前面我们说过LinkedList实现了链表、栈、队列等诸多数据结构,那么对这些数据结构的操作都对应了LInkedList类中的那些方法呢?来看下面的一个表格

public boolean add(E e)
在链表尾部插入元素
public void add(int index, E element)
在链表指定索引位置插入元素
public void addFirst(E e)
在链表头部插入元素
public void addLast(E e)
在链表尾部插入元素
public boolean addAll(Collection c)
在尾部追加新集合元素
public boolean addAll(int index, Collection c)
在指定索引位置追加集合中的元素
public void push(E e)

对于栈:将元素入栈

对于链表:头部插入与元素

public E pop()

对于栈:将栈顶元素弹出返回

对于链表:返回并删除链表头元素

public E pollFirst()
检索返回并删除链表第一个元素,如果链表为空返回null
public E pollLast()
检索返回并删除链表最后一个元素,如果链表为空返回null
public E peek()
检索返回但不删除链表第一个元素,如果链表为空则返回null
public E peekFirst()
检索返回但不删除链表第一个元素,如果链表为空则返回null
public E peekLast()
检索返回但不删除链表最后一个元素,如果链表为空则返回null
public E element()
检索但不删除此列表的头
public E set(int index, E element)
将指定索引位置的元素修改
public boolean offer(E e)
在链表尾部添加元素
public boolean offerFirst(E e)
在链表头部添加元素
public boolean offerLast(E e)
在链表尾部添加元素
public E get(int index)
返回链表中指定索引的元素
public E getFirst()
返回链表第一个元素
public E getLast()
返回链表中最后一个元素
public E remove()
检索返回并删除链表第一个元素,如果链表为空抛出
NoSuchElementException
public E remove(int index)
删除指定索引位置的元素,并将该元素返回
public boolean remove(Object o)
删除在该链表中第一次出现该元素位置的元素
public E removeFirst()

删除并返回链表中的第一个元素

Throws:
NoSuchElementException – if this list is empty

public E removeLast()

删除并返回链表中最后一个元素

Throws:
NoSuchElementException – if this list is empty

public boolean removeFirstOccurrence(Object o)
删除链表中第一次出现的元素O
public boolean removeLastOccurrence(Object o)
删除链表中最后一次出现的元素O

手写LinkedList加深印象 

下面我们自己尝试手写一个简单的LinkedList。由于底层是链表,如果学过数据结构,我相信即使不看源码自己也能手写一个能够储存若干数据的链表,并完成增删改查等功能。按照ArrayList分析思路,最好的思路是看一下LinkedList底层源码,并自己手写一个简单的LinkedList,这里源码不在贴出,手写的LinkedList仅仅实现了小部分功能,仅供参考

//带头尾指针的双向链表
public class MyLinkedList {
    public MyLinkedList() {
    }

    ;

    //定义数据结点
    private class Node {
        Node previous;//指向前一个结点的指针
        Node next;//指向后一个结点的指针
        E element;//储存数据的变量空间

        public Node(Node previous, Node next, E element) {
            super();
            this.element = element;
            this.next = next;
            this.previous = previous;
        }

        public Node(E element) {
            super();
            this.element = element;
        }
    }

    private Node first;//链表的头指针
    private Node last;//链表的尾指针
    private int size;

    public void add(Object o) {
        Node node = new Node(o);
        //如果第一个节点为空。即第一次增加节点
        if (first == null) {
            node.previous = null;
            node.next = null;
            first = node;
            last = node;
        } else {//如果不是第一次增加节点
            node.previous = last;//将原来链表的尾指针指向的结点设置为新增结点的前驱
            node.next = null;//新结点的后继为空
            last.next = node;//将原来链表的尾指针指向的结点的后继设置为新增结点
            last = node;//更新链表尾指针指向新增结点
        }
        size++;
    }

    public Object get(int index) {
        //检查索引是否合法
        checkIndex(index);
        //获取指定索引位置的结点对象
        Node node=getNodeByIndex(index);
        return node!=null?node.element:null;
    }

    public int size() {
        return size;
    }

    public void remove(int index) {
        checkIndex(index);
        Node node=getNodeByIndex(index);
        if(node!=null){
            Node prior=node.previous;
            Node next=node.next;
            if (prior!=null){
                //将要删除的结点的前驱结点的后继指针指向要删结点的后继结点
                prior.next=next;
            }else {
                //如果要删除的结点前驱为空,证明要删除的头节点,此时移动头指针
                first=first.next;
            }
            if (next!=null){
                //将要删除的结点的后继结点的前驱指针指向要删结点的前驱结点
                next.previous=prior;
            }else {
                //如果要删除的结点后继为空,证明要删除的尾节点,此时移动尾指针
                last=last.previous;
            }
        }
        size--;
    }

    public void checkIndex(int index) {
        if (index < 0 || index > size - 1) {
            throw new RuntimeException("非法的索引:" + index);
        }
    }

    public Node getNodeByIndex(int index) {
        if(size==0){
            return null;
        }
        Node temp = first;
        for (int i = 0; i < index; i++) {
            temp = temp.next;
        }
        return temp;
    }
}

LinkedList集合中迭代器的fail-fast机制

有关迭代器的中的fail-fast机制及其原理,请查看本人写的另外一篇文章

你可能感兴趣的:(Java面试专栏,java)