本文基于jdk1.8_171

LinkedList介绍

之前看了ArrayList,内部是一个数组。这次看了LinkedList,作用和ArrayList一样,但是内部是链表形式。链表结构如下图:

数组和链表的区别

直接看例子:

数组: 假设有10个人去看电影,想要挨着坐,那就需要找同一排连续的10个位置坐(座位号01-10)。如果想要找5号先生,那直接去第五个座位即可。但是,如果此时来了第十一个人,想要坐在4号和5号先生的位置中间,那么原本5号先生和后面的所有人都要往后挪个位置,然后11号先生才能落座。

链表: 还是10个人去看电影,不用挨着坐,每个人记着自己前一个人和后一个人坐哪里就行,第一个人记下一个人和最后一个人记上一个人。此时还是来了第十一个人,想要坐在4号和5号先生的位置中间(不是空间意义的中间),那么5号先生更换自己上家为11号,4号先生更换下家为11号,11号记自己上家是4号,下家是5号即可。但是如果需要找到5号的位置,那你需要先找1号,问他下家2号在哪里,然后找2号下家3号。。。一直找到5号。

所以结论:数组查找元素比链表快,但往中间插入元素链表比数组快。

内部类

/**
* 链表内部元素
*/
class Node<E>{
//当前元素
E item;
//上一个元素
Node<E> prev;
//下一个元素
Node<E> next;
Node(Node<E> prev,E element,Node<E> next){
this.item = element;
this.next = next;
this.prev = prev;
}
}

成员变量

//元素个数
transient int size = 0;
//链表第一个元素
transient Node<E> first;
//链表最后一个元素
transient Node<E> last;

常用方法

1、addFirst(E e)

public void addFirst(E e){
linkFirst(e);
} private void linkFirst(E e){
//缓存第一个元素
final Node<E> f = first;
//新建元素
final Node<E> newNode = new Node<>(null, e, f);
//第一个元素设置为新元素
first = newNode;
if(f == null)
//如果链表原先是空的,那添加元素后最后一个元素也是新元素
last = newNode();
else
//如果链表原先不为空,那么原先第一个元素的前一位指向新元素
f.prev = newNode;
//链表元素个数加一
size++;
//集合修改次数加一
modCount++;
}

addLast类似,不赘述。

2、add(E e)

public boolean add(E e){
//add方法就是在链表末尾添加元素
linkLast(e);
return true;
}

3、add(int index,E element)

public void add(int index,E element){
//这一步就是检查index是否越界,越界就抛出异常
checkElementIndex(index);
//如果index是size,就加在链表最后,否则就加在指定元素之前
if(index == size)
linkLast(element);
else
//node(index)方法解析往下看
linkBefore(element,node(index));
} 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的下一个元素变成新元素
pred.next = newNode;
}
size++;
modCount++;
}

4、get(int index)

public E get(int index){
checkElementIndex(index);
//node(index)方法解析往下看
return node(index).item;
} Node<E> node(int index){
if(index < (size >> 1)){
//如果index小于size的一半,则从头开始找
//先找到第一个元素
Node<E> x = first;
//往后找index次,我看的时候第一次没理解,举了例才明白,比如index=1,size=4等等
for(int i=0; i < index; i++)
x = x.next;
return x;
} else {
//从尾开始找
//先找到最后一个元素
Node<E> x = last;
//往前找index次
for(int i = size - 1; i > index; i--)
x = x.prev;
return x;
}
}

结语

LinkedList有自身的优点,虽然实际运用不如ArrayList频繁,但我们依然要了解它的特性以及实现方式。

jdk源码--LinkedList的更多相关文章

  1. JDK源码分析(三)—— LinkedList

    参考文档 JDK源码分析(4)之 LinkedList 相关

  2. JDK源码分析(2)LinkedList

    JDK版本 LinkedList简介 LinkedList 是一个继承于AbstractSequentialList的双向链表.它也可以被当作堆栈.队列或双端队列进行操作. LinkedList 实现 ...

  3. JDK源码学习LinkedList

    LinkedList是List接口的子类,它底层数据结构是双向循环链表.LinkedList还实现了Deque接口(double-end-queue双端队列,线性collection,支持在两端插入和 ...

  4. 一点一点看JDK源码(六)java.util.LinkedList前篇之链表概要

    一点一点看JDK源码(六)java.util.LinkedList前篇之链表概要 liuyuhang原创,未经允许禁止转载 本文举例使用的是JDK8的API 目录:一点一点看JDK源码(〇) 1.什么 ...

  5. JDK源码学习系列05----LinkedList

                                             JDK源码学习系列05----LinkedList 1.LinkedList简介 LinkedList是基于双向链表实 ...

  6. 【java基础之jdk源码】集合类

    最近在整理JAVA 基础知识,从jdk源码入手,今天就jdk中 java.util包下集合类进行理解 先看图 从类图结构可以了解 java.util包下的2个大类: 1.Collecton:可以理解为 ...

  7. Java中集合框架,Collection接口、Set接口、List接口、Map接口,已经常用的它们的实现类,简单的JDK源码分析底层实现

    (一)集合框架: Java语言的设计者对常用的数据结构和算法做了一些规范(接口)和实现(实现接口的类).所有抽象出来的数据结构和操作(算法)统称为集合框架. 程序员在具体应用的时候,不必考虑数据结构和 ...

  8. JDK源码阅读顺序

      很多java开发的小伙伴都会阅读jdk源码,然而确不知道应该从哪读起.以下为小编整理的通常所需阅读的源码范围. 标题为包名,后面序号为优先级1-4,优先级递减 1.java.lang 1) Obj ...

  9. 一点一点看JDK源码(〇)

    一点一点看JDK源码(〇) liuyuhang原创,未经允许进制转载 写在前面: 几乎所有的大神都会强调看源码,也强调源码的重要性: 但是如何看源码,源码看什么?看了什么用?看了怎么用? 困扰很多人, ...

随机推荐

  1. PHPEXCEL 不能输出中文内容,只显示空白

    以他带的示例文件为例 01simple-download-xls.php // Add some data $objPHPExcel->setActiveSheetIndex(0)        ...

  2. Android零基础入门第21节:ToggleButton和Switch使用大全

    原文:Android零基础入门第21节:ToggleButton和Switch使用大全 上期学习了CheckBox和RadioButton,那么本期来学习Button的另外两个子控件ToggleBut ...

  3. 硬盘可以支持140万小时(也就是159年)的MTBF(硬盘只是一次性的投入)

    1.硬盘的的确确是一个一次性投入: 最普通的家用硬盘寿命都可以到达平均5年以上:企业级的硬盘的寿命更是长的离谱,如这个西数为数据中心提供的硬盘: WD Re:页面上说明该种硬盘可以支持140万小时(也 ...

  4. DataTable,DataView 排序和使用

    我们都知道在Sql Server可以用order by来排序,所以很多朋友在DataTable中排序也想到了用order by关键字.但这样实现是比较困难的,下面,我们讲解一种比较简单的方法: 控制台 ...

  5. Lua中的协同程序 coroutine(转)

    Lua中的协程和多线程很相似,每一个协程有自己的堆栈,自己的局部变量,可以通过yield-resume实现在协程间的切换.不同之处是:Lua协程是非抢占式的多线程,必须手动在不同的协程间切换,且同一时 ...

  6. Q_DECLARE_METATYPE(继承QObject的类都已经自动注册),注册后的类型可以作为QVariant的自定义类型

    简介 这个宏用来注册一个类(含默认构造.默认析构.拷贝构造函数)为QMetaType类型 ,注册后的类型可以作为QVariant的自定义类型. 这个宏应该放在类或者结构体外面的下面,也可以放在一个非公 ...

  7. javascript学习路线图

    史上最全的javascript学习路线图 JavaSctipt学习路线 完成整个课程大纲需要花上6~8周的时间,将学会完整的JavaScript语言(包括jQuery和一些HTML5).如果你没有时间 ...

  8. 代理Delegate的小应用(使用setModelData设置下拉日期对话框)

    前言 在平时关于表格一类的的控件使用中,不可避免需要修改每个Item的值,通过在Item中嵌入不同的控件对编辑的内容进行限定,然而在表格的Item中插入的控件始终显示,当表格中item项很多的时候,会 ...

  9. SAP HANA 开发模式 - 基于SAP HANA平台的多团队产品研发

    “基本”开发模式 Windows: Unix/Linux: 在基本模式下我们可以通过regi来进行激活我们的object.Regi是一个类git功能的,方便和HANA repository交互的一个命 ...

  10. chrome和safari字体粗细问题

    因为我用的是mac电脑,写项目所遇到的问题,这也是我上网和手动试了多次,觉得有效,分享给大家 -webkit-font-smoothing: subpixel-antialiased; -webkit ...