赞
踩
目录
LinkedList 是一个双向链表结构(JDK1.6 之前为循环链表,JDK1.7 取消了循环),在任意位置插入删除都很方便,但是不支持随机取值,每次都只能从一端开始遍历,直到找到查询的对象,然后返回;不过,它不像 ArrayList 那样需要进行内存拷贝,因此相对来说效率较高,但是因为存在额外的前驱和后继节点指针,因此占用的内存比 ArrayList 多一些。
LinkedList 采用链表存储,所以对于add(E e)方法的插入,删除元素时间复杂度不受元素位置的影响,近似 O(1),如果是要在指定位置i插入和删除元素的话((add(int index, Eelement)) 时间复杂度近似为o(n))因为需要先移动到指定位置再插入
如下图:
观察上图:
- public LinkedList() {
- }
总结:无参构造方法,此时双向链表的节点数量size为0,双向链表的头尾节点为null。
- public LinkedList(Collection<? extends E> c) {
- this();
- addAll(c);
- }
总结:先将集合转换为数组,然后将数组中的元素按照索引顺序一个个从双向链表的尾部插入到空的双向链表中。
- private static class Node<E> {
- //元素
- E item;
- //后驱指针
- Node<E> next;
- //前驱指针
- Node<E> prev;
- //构造方法
- Node(Node<E> prev, E element, Node<E> next) {
- this.item = element;
- this.next = next;
- this.prev = prev;
- }
- }
- public boolean add(E e) {
- //将元素添加到链表的尾部
- linkLast(e);
- return true;
- }
- void linkLast(E e) {
- final Node<E> l = last;
- //创建新节点
- final Node<E> newNode = new Node<>(l, e, null);
- //把之前的尾指针节点指向新节点
- last = newNode;
- //如果尾节点为空,则代表是新链表,直接赋值给头指针;如果不为空,则把尾指针指向新节点
- if (l == null)
- first = newNode;
- else
- l.next = newNode;
- //长度++
- size++;
- //代表对集合的操作次数
- modCount++;
- }

通过上述分析,可以看到:add()方法调用了linkLast()方法,将新节点插入到链表的尾部。在linkLast()方法里对尾指针last进行了判断,如果尾节点为空,说明是第一次插入元素,则直接将新节点赋值给头指针;如果尾节点不为空,则将节点的尾指针指向新节点即可,然后再将size和modCount自增1。modCount不是LinkedList里的变量,而是来自于AbstractList。
接下来看一下如何在指定位置添加元素:
- public void add(int index, E element) {
- //检查索引位置
- checkPositionIndex(index);
- //如果和当前长度size相等,则直接添加元素到末尾,否则就将元素插入到指定的位置
- if (index == size)
- linkLast(element);
- else
- linkBefore(element, node(index)); //node用来获取给定index处的元素节点
- }
- //索引校验
- private void checkPositionIndex(int index) {
- if (!isPositionIndex(index))
- throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
- }
- private boolean isPositionIndex(int index) {
- return index >= 0 && index <= size;
- }
- //linkLast()前面已经介绍,这里只展示linkBefore()
- void linkBefore(E e, Node<E> succ) {
- // assert succ != null;
- final Node<E> pred = succ.prev;
- final Node<E> newNode = new Node<>(pred, e, succ);
- succ.prev = newNode;
- if (pred == null)
- first = newNode;
- else
- pred.next = newNode;
- size++;
- modCount++;
- }

由上述分析可以看到:在指定位置添加元素的时候,首先调用checkPositionIndex()方法判断下标是否越界,然后判断index是否等于 size,如果相等则添加到末尾,否则将该元素插入的 index 的位置。linkBefore()方法负责把元素 e 插入到 succ 之前。
node(index)方法是获取 index 位置的节点,它将index与当前链表的一半进行比较,如果比一半小则从头遍历,如果比一半大则向后遍历。
- 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;
- }
- }
在前面介绍LinkedList的有参构造时,我们可以看到其调用了addAll()方法,那么接下来看看这个方法又是如何实现的?
- //调用addAll(index, c)
- public boolean addAll(Collection<? extends E> c) {
- return addAll(size, c);
- }
- public boolean addAll(int index, Collection<? extends E> c) {
- //检查index是否越界
- checkPositionIndex(index);
- //将集合转为数组
- Object[] a = c.toArray();
- int numNew = a.length;
- if (numNew == 0)
- return false;
-
- Node<E> 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<E> 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的值
- //操作次数modCount+1
- size += numNew;
- modCount++;
- return true;
- }

如果是删除指定位置的元素,则先检查下标是否越界,然后再调用unlink()方法释放节点,移除掉指定的元素。
- //移除指定位置的元素
- public E remove(int index) {
- checkElementIndex(index);
- return unlink(node(index));
- }
- E unlink(Node<E> x) {
- // assert x != null;
- final E element = x.item;
- final Node<E> next = x.next;
- final Node<E> prev = x.prev;
- //如果移除的是头节点,则头节点后移
- if (prev == null) {
- first = next;
- } else {
- prev.next = next; //否则就释放节点的迁移借宿
- x.prev = null;
- }
- //如果溢出的是尾节点,则尾节点前移
- if (next == null) {
- last = prev;
- } else {
- next.prev = prev; //否则就释放节点的后一个元素
- x.next = null;
- }
- //节点数据置为空
- x.item = null;
- size--;
- modCount++;
- return element;
- }

remove(Object o)从该列表中删除第一个出现的指定元素(如果存在)。如果此列表不包含该元素,则它将保持不变。
- public boolean remove(Object o) {
- if (o == null) {
- for (Node<E> x = first; x != null; x = x.next) {
- if (x.item == null) {
- unlink(x);
- return true;
- }
- }
- } else {
- for (Node<E> x = first; x != null; x = x.next) {
- if (o.equals(x.item)) {
- unlink(x);
- return true;
- }
- }
- }
- return false;
- }

除了上述的移除元素方法,还有一些其他的方法,如:removeFirst()、removeLast()、poll()等,这里不再赘述。
set()方法通过修改下标获取到下标节点,获取出旧值返回,把新值赋值元素
- public E set(int index, E element) {
- //检查索引是否越界
- checkElementIndex(index);
- //获取到要修改的元素的下标
- Node<E> x = node(index);
- //获取旧值
- E oldVal = x.item;
- //修改
- x.item = element;
- return oldVal;
- }
get()方法根据下标获取元素遍历找到当前元素并返回,遍历利用的是判断当前获取元素位于链表的前半段还是后半段,前半段则从头遍历到当前位置返回,后半段则从尾遍历到当前位置返回
- public E get(int index) {
- checkElementIndex(index);
- return node(index).item;
- }
除此之外,还有getFirst()、element()、peek()、peekFirst() 这四个获取头结点方法,区别在于对链表为空时的处理,是抛出异常还是返回null,其中getFirst() 和element() 方法将会在链表为空时,抛出异常;因为内部都保存了头节点所以直接获取头节点就可以。getLast() 方法在链表为空时,会抛出NoSuchElementException,而peekLast() 则不会,只是会返回 null;内部保存了尾节点直接返回即可。
双向链表就是一个元素有3个属性,一个向前的指针,一个向后的指针,一个当前节点值;双向就是本节点既有向后的指向,也有向前的
双向循环链表的差别在于循环,双向链表首位不相连,指针都指向空,双向循环链表是首位相连形成环状
双向循环链表是通过new一个headerEntry管理首尾相连得,可以少创建对象
写操作主要分为2种,一种头尾插入,一种中间插入;双向链表的有点在于头尾插入的时候只需要维护一个指针,中间插入2个没什么区别,但实际使用中头尾插入是最频繁的
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。