• 一、 概述
    • 二、 set和get函数 {#2-_set和get函数}

    一、 概述

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

    按下标访问元素—get(i)/set(i,e) 要悲剧的遍历链表将指针移动到位(如果i>数组大小的一半,会从末尾移起)。

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

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

    Doubly-linked list implementation of the List and Deque interfaces. Implements all optional list operations, and permits all elements (including null).

    1. LinkedList<String> list = new LinkedList<String>();
    2. list.add("语文: 1");
    3. list.add("数学: 2");
    4. list.add("英语: 3");

    结构也相对简单一些,如下图所示:
    Java集合——LinkedList - 图1

    二、 set和get函数 {#2-_set和get函数}

    1. public E set(int index, E element) {
    2. checkElementIndex(index);
    3. Node<E> x = node(index);
    4. E oldVal = x.item;
    5. x.item = element;
    6. return oldVal;
    7. }
    8. public E get(int index) {
    9. checkElementIndex(index);
    10. return node(index).item;
    11. }

    这两个函数都调用了node函数,该函数会以O(n/2)的性能去获取一个节点,具体实现如下所示:

    1. Node<E> node(int index) {
    2. // assert isElementIndex(index);
    3. if (index < (size >> 1)) {
    4. Node<E> x = first;
    5. for (int i = 0; i < index; i++)
    6. x = x.next;
    7. return x;
    8. } else {
    9. Node<E> x = last;
    10. for (int i = size - 1; i > index; i--)
    11. x = x.prev;
    12. return x;
    13. }
    14. }

    就是判断index是在前半区间还是后半区间,如果在前半区间就从head搜索,而在后半区间就从tail搜索。而不是一直从头到尾的搜索。如此设计,将节点访问的复杂度由O(n)变为O(n/2)。