LinkedList

概述

  以双向链表实现,链表没有容量的限制,但双向链表本身使用了更多的空间,也需要额外的链表指针操作;

  按照下标访问元素--get(i)/set(i,e)要遍历链表将指针移动到相应位置(如果i》数组大小的一半,会从末尾移动);

  插入、删除元素时修改前后节点的指针即可,但还是要遍历部分链表的指针才能移动到下标所指的位置,只有在链表两头的操作--add()、addFirst()、removeLast(),或者用iterator()上的remove()能省掉指针的移动;

  LinkedList是一个简单的数据结构,与ArrayList不同的是,它是基于链表实现的。

set()和get()方法

public E set(int index, E element) {
    checkElementIndex(index);
    Node<E> x = node(index);
    E oldVal = x.item;
    x.item = element;
    return oldVal;
}

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

private void checkElementIndex(int index) {
    if (!isElementIndex(index))
        throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
}

  这两个方法都调用了node()方法,这个方法会以O(n/2)的性能去获取一个节点。如下:

Node<E> node(int index) {
    // assert isElementIndex(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;
    }
}

  该方法就是判断index是在前半区间还是后半区间,如果在前半区间就从头部搜索,而在后半区间就从尾部搜索。而不是一直从头到尾的搜索。这样子做,将节点访问的复杂度由O(n)变为了O(n/2)。

-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-----------------last line for now-----------------