从源码分析java集合类原理(2)-LinkedList原理分析

在介绍LinkedList之前我们先来简单介绍一下链表这种数据结构,与顺序表不同,链表不限制数据的物理存储状态,换句话说,使用链表存储的数据元素,其物理存储位置是随机的。链表存储数据的单元叫做节点,每个节点的存储都包括数据元素本身,其所在的区域称为数据域,指向直接后继元素的指针,所在的区域称为指针域;链表的基本结构如下图所示


单向链表

上面讲到的链表每个节点指针域都是指向后面一个节点,也就是直接后继节点,这类链表叫做单向链表或者单链表,单向链表有一个天生的缺点就是无法查找指定节点的前驱节点,为了解决这个问题,双向链表应运而生。双向链表与单向链表相比,多了一个指向前驱节点的指针域,该指针直接指向前驱节点,双向链表的基本结构如下图所示。


双向链表

从上图可以看出,双向链表每一个节点包括三个部分,前驱指针域,数据本身,后继指针域,同时还存在始终指向头结点的HEAD指针和始终指向尾节点的LAST指针。本文所讲到的LinkedList集合类就是采用双向链表的数据结构来存储数据的。

那么链表跟ArrayList采用的数组线性表相比有什么优缺点呢?

那么链表跟ArrayList采用的数组线性表相比有什么优缺点呢?

    1、数组线性表需要一块连续的内存存储空间来存储数据,而链表由于指针的存在,不需要存储空间连续,只要有存储空间,数据可随意存取。

    2、数组线性表查找数据快,直接通过下标可直接获取;而链表需要遍历整个链表,效率相对较低。

    3、数组线性表添加和删除需要移动数组元素,速度相对慢;而链表添加删除数据只需要改变指针指向就可以了,速度相对快。

    4、数组线性表一旦定义长度无法动态增长;链表存储是在程序运行过程中动态的分配空间

    5、链表需要额外的空间来存储指针。



分析完链表数据结构,下面我们结合源码来分析下LinkedList到底是如何实现数据存取操作的。

首先来看数据节点的定义,结合上面的所讲的双向链表的结构,定义的Node类为链表的节点,item为数据本身,next为指向后继节点指针,prev为指向前驱节点指针。


节点定义

LinkedList共有两个构造函数,无参构造函数生成一个空的链表,带集合类构造函数将指定集合类数据构造到链表。


构造函数

接下来我们继续看下LinkedList的数据操作

(1)、添加数据。


添加数据


添加数据

Add方法默认将数据添加到链表末尾,在添加新元素数据时,首先创建一个节点用来保存数据,该节点的前驱节点指向当前的尾节点,该节点的后继节点指向NULL,当前尾节点的后继节点指向新创建的节点,具体过程简化为下图。


(2)、添加数据到指定位置。上面我们讲到的add方法是将新数据添加到链表末尾,那么如何将新数据添加到指定位置呢,继续看源码。


添加数据到指定位置

源码中首先会去检查位置的有效性,下标必须在0到当前链表大小范围之内才是有效的位置。


校验下标

接下来开始添加数据,如果下标为链表大小,相当于需要将新数据添加到链表末尾,处理方式就是我们在(1)中讲的过程,这里不再赘述;如果在链表中间位置添加数据,首先会调用node(int index)方法,该方法通过遍历链表获取添加数据之前index位置的节点,那么如何遍历呢?这里不是纯粹的从头节点开始遍历,而是首先判断index位置,如果是在链表的前半部分则从头结点开始遍历,如果index位置在链表的后半部分,则开始从尾节点向前遍历,这样大大节省了遍历的效率,这里拿到的节点我们称作老节点。具体源码如下:


获取指定位置节点

获取到index位置的节点之后,开始向链表中添加数据,具体操作就是首先创建一个新数据节点,新节点的前驱节点为老节点前驱节点,新节点的后继节点为老节点,同时更新老节点前驱节点的后继节点为新节点,更新老节点的前驱节点为新节点,文字描述起来有点绕,下面为具体源码及详细过程。


添加节点


添加节点详解

(3)、获取指定位置元素。获取指定位置元素其实在上面的(2)中已经提到过,就是node(int index)方法的实现,这里的获取跟上面的逻辑完全一样,这里就不在赘述。


get方法

(4)、删除指定位置元素。经过以上对添加数据原理的深入了解,删除元素的具体逻辑其实可以对照来看,首先是检查下标的有效性,然后通过node(int index)方法找到指定位置的节点,也就是删除的目标节点,接下来就是对指针域的操作,如果目标节点的前驱节点为空,则说明目标节点就是首节点,那么直接将目标的节点的后继节点直接当做首节点就可以了;如果目标节点不是首节点,则将目标节点的后继节点作为目标节点前驱节点的后继节点就可以了;如果目标节点的后继节点为空,则说明目标节点为尾节点,则直接将目标节点的前驱节点作为尾节点就可以了;如果目标节点不是尾节点,则将目标节点的前驱节点作为目标节点后继节点的前驱节点就可以了。同样我们将过程通过图的形式表现出来更容易理解一些。


remove方法


删除节点


删除节点详解

(5)、删除指定元素。删除指定元素的中心思想还是通过遍历链表找到目标元素的下标位置,然后通过上面讲到的通过删除指定位置的元素来进行删除。


删除指定元素



总结:LinkedList采用的是双向链表的数据结构来存取数据的,与ArrayList相比,LinkedList有更快的添加和删除操作,但是读取数据的速度相对ArrayList来说相对较慢,这里的快慢一定是针对大数据量而言的。另外LinkedList的所有方法与ArrayList一样,都是非线程安全的,在多线程环境下使用,需要我们进行数据操作的同步控制。

才疏学浅,码字不易,有总结不到位的请各路大神多多指教,欢迎交流!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342