【数据结构】2.java源码关于LinkedList

关于LinkedList的源码关注点


1.从底层数据结构,扩容策略
2.LinkedList的增删改查
3.特殊处理重点关注
4.遍历的速度,随机访问和iterator访问效率对比

 

1.从底层数据结构,扩容策略

构造函数不做任何操作,只要再add的时候进行数据初始化操作,以操作推动逻辑,而且linkedlist是一个双向链表,所以可以向前向后双向遍历
由于构造函数并没有任何操作,其实这里我们可以先看新增操作,并且因为用的是链表所以无法随机访问,这里随机读取就会比较慢

?

分享图片

 


底层结构就是size,首节点,尾节点,还有就是一个List都有的共性就是modCount,这值用来记录这个list被修改了多少次

 

 2.LinkedList的增删改查

 2.1 add操作,linklast

 

Add操作的实质就是进行linklast操作

linklast的操作就是再最后吧节点添加到尾部,并修正size大小

 

分享图片

 

 

    public boolean@H_502_58@ add(E ele) {
        linkLast(ele);
        return true@H_502_58@;
    }

 

我们看看如果是在指定的位置插入元素的操作
首先要确认index再指定范围内
这里有个小优化,如果是在末尾进行添加的话,我们直接调用linklast就可以了
如果不是最后一个,那么首先要获取指定位置的node节点,我们遍历指定位置的时候
可以确定index的位置如果过半了,那么就从后往前,如果没有过半,那么就从前往后
1.直接再index创建一个节点,然后前后合并一下
2.吧原来的index位置的节点断开,吧这个节点的pre指向新节点
3.判断前置节点是否到头了,为空
4.把前置节点的next指向新节点

 先看看node定位操作

 

分享图片

 

 然后我们看看linkbefore,前置插入

 

public void linkBefore(E e,Node<E>@H_502_58@ succ) {
        final Node<E> pred =@H_502_58@ succ.prev;
        final Node<E> newNode = new Node<>@H_502_58@(pred,e,succ);
        succ.prev =@H_502_58@ newNode;
        if (pred == null@H_502_58@)
            first =@H_502_58@ newNode;
        else@H_502_58@
            pred.next =@H_502_58@ newNode;
        size++@H_502_58@;
        modCount++@H_502_58@;
    }

那么我们需要插入到指定的位置的方法就可以很简单的实现了
Linkbefore(e,node(index))即可

 

分享图片

2.2 删除

 

查看源码,比较remove(),remove(object),remove(index)等等操作,归根到底还是一个unlink操作
我们需要对指定的节点进行unlink操作
就2步操作
1.当前节点的前一个节点指向当前节点的下一个节点,说白了node.pre.next = node.next;
2.当前节点的下一个节点的前置节点指向当前节点的上一个节点:node.next.pre = node.pre;
其他细节部分就是首尾节点的处理

 

 

public E unlink(Node<E>@H_502_58@ node) {
        // assert x != null;
        //这里需要操作的就是三个节点,前置节点,当前节点,后置节点
        final E element =@H_502_58@ node.item;
        final Node<E> prev =@H_502_58@ node.prev;
        final Node<E> next =@H_502_58@ node.next;

        //当前节点的前一个节点指向当前节点的下一个节点,说白了node.pre.next = node.next;
        if (prev == null@H_502_58@) {
            //避免首节点操作
            first =@H_502_58@ next;
        } else@H_502_58@ {
            prev.next =@H_502_58@ next;
            node.prev = null; //断开原始连接
@H_502_58@        }

        //当前节点的下一个节点的前置节点指向当前节点的上一个节点:node.next.pre = node.pre;
        if (next == null@H_502_58@) {
            last =@H_502_58@ prev;
        } else@H_502_58@ {
            next.prev =@H_502_58@ prev;
            node.next = null@H_502_58@;
        }

        //清除当前节点
        node.item = null@H_502_58@;
        size--@H_502_58@;
        modCount++@H_502_58@;
        return@H_502_58@ element;
    }

其余操作基本就是调用unlink方法进行操作

分享图片

 

 修改set和获取get就不多说了,就注意一点就是set操作的时候,会返回旧值,并且这两个操作不会修改modCount值,也就是不会产生链表变动

 

 3.特殊处理重点关注,iterator,序列化

 

 

对于iterator这个就不多说了,其实还是调用上面的那些方法,遍历也就是next和pre和循环遍历没差别
但是注意一点就是通过迭代器进行remove和add是可以的,但是注意一点,如果使用迭代器进行remove或者add操作的通过,还使用了一般的remove和add那么就会使迭代器失效


序列化这里我们先只做一个了解,后续开章节专门学习一下java的序列化:

进行序列化、反序列化时,虚拟机会首先试图调用对象里的writeObject和readObject方法,进行用户自定义的序列化和反序列化。如果没有这样的方法,那么默认调用的是ObjectOutputStream的defaultWriteObject以及ObjectInputStream的defaultReadObject方法。换言之,利用自定义的writeObject方法和readObject方法用户可以自己控制序列化和反序列化的过程。
---------------------
版权声明:本文为CSDN博主「zthgreat」的原创文章,遵循CC 4.0 by-sa版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/u014634338/article/details/78165127

 

 

4.遍历的速度,随机访问和iterator访问效率对比

这个问题其实也可以舍弃掉了,这里遍历的实质还是使用链表的遍历方式进行遍历

 

 

5.是否支持多线程

LinkedList 是线程不安全的,允许元素为null的双向链表。

 

参考:https://blog.csdn.net/u014634338/article/details/78165127

相关文章

键树的基本概念 键树又称数字查找树(Digital Search Tree)。 它是一棵度大于等于2的树,树中的每个结...
[TOC] 基本概念 数据: 数据 是对客观事物的符号表示,在计算机科学中指所有能输入到计算机中并被计算机...
[TOC] 反证法 基本概念: 一般地,假设原命题不成立(即 在原命题的条件下,结论不成立),经过正确的推...
最近抽空整理了&quot;数据结构和算法&quot;的相关文章。在整理过程中,对于每种数据结构和算法...
[TOC] 矩阵在计算机图形学、工程计算中占有举足轻重的地位。在数据结构中考虑的是如何用最小的内存空间...
[TOC] 大O表示法:算法的时间复杂度通常用大O符号表述,定义为T[n] = O(f(n))。称函数T(n)以f(n)为界或...