data:image/s3,"s3://crabby-images/17bde/17bde8872a3cfaf4dd9117661edbd99301f973f5" alt="Java修炼指南:高频源码解析"
2.3 List集合的另一种典型实现——LinkedList类
上一节介绍了List集合的一种典型实现ArrayList,明白了ArrayList是由数组构成的,本节介绍List集合的另一种典型实现——LinkedList类,这是一个由链表构成的类。
链表(Linked list)是一种常见的基础数据结构,是一种线性表,但是它并不会按线性的顺序存储数据,而是在每一个节点里存储到下一个节点的指针(Pointer)。
使用链表结构可以克服数组链表需要预先知道数据大小的缺点,链表结构可以充分利用计算机内存空间,实现灵活的内存动态管理。但是链表失去了数组随机读取数据的优点,同时链表由于增加了节点的指针域,空间开销比较大。
了解完链表后,下面给出LinkedList的定义。
2.3.1 LinkedList定义
LinkedList是一个用链表实现的集合,元素有序且可以重复。LinkedList类结构如图2-3所示。
data:image/s3,"s3://crabby-images/59678/5967865a241fdfa9c8bdfabe5867d161b3168804" alt=""
和ArrayList集合一样,LinkedList集合也实现了Cloneable接口和Serializable接口,分别用来支持克隆以及支持序列化。List接口则定义了一套List集合类型的方法规范。
注意:
相对于ArrayList集合,LinkedList集合多了一个Deque接口,这是一个双向队列接口,双向队列就是两端都可以进行增加和删除操作。
data:image/s3,"s3://crabby-images/5a443/5a44344c041f2850b4c66c82700e28567b86e7e7" alt=""
●图2-3 LinkedList类结构图
2.3.2 字段属性
字段属性代码显示如下:
data:image/s3,"s3://crabby-images/322c4/322c4fc4e0966a6eb828506da1310a290103c642" alt=""
注意:
这里出现了一个Node类,这是LinkedList类中的一个内部类,其中每一个元素代表一个Node类对象,LinkedList集合就是由许多个Node对象类似于手拉着手构成,如图2-4所示。
data:image/s3,"s3://crabby-images/c5ced/c5ced6a3c11b28e7ab374dcd128bb6cfcdc8ffc6" alt=""
data:image/s3,"s3://crabby-images/6b5e2/6b5e2f27d60bc291a4040e12e3203ef8b7654380" alt=""
data:image/s3,"s3://crabby-images/9c56f/9c56fb73a6fcc69207a6820e87daedb16705cf4b" alt=""
●图2-4 LinkedList结构图
上图的LinkedList有四个元素,也就是由4个Node对象组成,size=4,head指向第一个elementA,tail指向最后一个节点elementD。
2.3.3 构造函数
LinkedList有两个构造函数,第一个是默认的空的构造函数,第二个是将已有元素的集合Collection的实例添加到LinkedList中,调用的是addAll()方法,这个方法下面介绍。
data:image/s3,"s3://crabby-images/9d772/9d7728184717c590c88feb95f6b704881daead7e" alt=""
注意:
LinkedList没有初始化链表大小的构造函数,因为链表不像数组,一个定义好的数组是必须先要有确定的大小,然后再去分配内存空间,而链表不一样,它没有确定的大小,通过指针的移动来指向下一个内存地址的分配。
2.3.4 添加元素
1. addFirst(E e)
将指定元素添加到链表头,如图2-5所示。
data:image/s3,"s3://crabby-images/e2c7b/e2c7ba243187f6d96afe87f00af5050ccf538482" alt=""
●图2-5 将指定元素添加到链表头
data:image/s3,"s3://crabby-images/e0d71/e0d71afb83e83427c1b68115b21509a321777bb5" alt=""
2. addLast(E e)和add(E e)
将指定元素添加到链表尾,如图2-6所示。
data:image/s3,"s3://crabby-images/bd554/bd554aa3a3fbbf083d3ea4d0001578dcc06d9c92" alt=""
●图2-6 将指定元素添加到链表尾
data:image/s3,"s3://crabby-images/67c37/67c3720bc09fbe12c9220016533cb4fbbcbf6107" alt=""
3. add(int index,E element)
将指定的元素插入此列表中的指定位置,如图2-7所示。
data:image/s3,"s3://crabby-images/145ee/145eed167a0bb50c152664befd7cb36038cd7661" alt=""
●图2-7 将指定元素插入到指定位置
data:image/s3,"s3://crabby-images/c2bb7/c2bb71491d075477000324b3cfee82056e7b0de2" alt=""
data:image/s3,"s3://crabby-images/5ca93/5ca93838b9befe62dc942aaa4c5303454bc47895" alt=""
2.3.5 删除元素
删除元素和添加元素一样,也是通过更改指向上一个节点和指向下一个节点的引用即可,这里就不作图形展示了。
1. remove()和removeFirst()
该命令为从此列表中移除并返回第一个元素。
data:image/s3,"s3://crabby-images/31229/312296b2a0e88edba2b6212efd3cae6970266960" alt=""
data:image/s3,"s3://crabby-images/666de/666de4e05e79ffd0849a016aa30b60f5e9d3cd7f" alt=""
2. removeLast()
该命令为从该列表中删除并返回最后一个元素。
data:image/s3,"s3://crabby-images/4943b/4943b22f491c906452c0f53ead48cab9df9201b2" alt=""
3. remove(int index)
该命令为删除此列表中指定位置的元素。
data:image/s3,"s3://crabby-images/167ce/167ced1900339c95cb4fedbf107b145fb7b7a443" alt=""
4. remove(Object o)
如果该命令存在,则表示从该列表中删除第一次出现的指定元素。
此方法本质上和remove(int index)没多大区别,通过循环判断元素进行删除,需要注意的是,是删除第一次出现的元素,不是所有的。
data:image/s3,"s3://crabby-images/530ec/530ecc139dfaaf6f9ff512decf5b260c828e1b7a" alt=""
2.3.6 修改元素
通过调用set(int index,E element)方法,用指定的元素替换此列表中指定位置的元素。
data:image/s3,"s3://crabby-images/c2923/c292347304349e02d4143ca1180bb75e6cb25623" alt=""
这里主要是通过node(index)方法获取指定索引位置的节点,然后修改此节点位置的元素即可。
2.3.7 查找元素
1. getFirst()
该命令为返回此列表中的第一个元素。
data:image/s3,"s3://crabby-images/40a14/40a143ca063d20e00d7e83f64e61c3d8305fadba" alt=""
2. getLast()
该命令为返回此列表中的最后一个元素。
data:image/s3,"s3://crabby-images/92123/921238d47935f43c576a41240bdeb5a7a3388930" alt=""
3. get(int index)
该命令为返回指定索引处的元素。
data:image/s3,"s3://crabby-images/a4aa3/a4aa3e528c9f030f5ae9a9df4fa127232f669b4f" alt=""
4. indexOf(Object o)
该命令为返回此列表中指定元素第一次出现的索引,如果此列表不包含元素,则返回-1。
data:image/s3,"s3://crabby-images/79c33/79c330bea38f561327b8d02383c0e0bb1f2b048a" alt=""
2.3.8 遍历集合
1. 普通for循环
data:image/s3,"s3://crabby-images/09e65/09e6506077a29fe76607261933bae1ac2c7099d2" alt=""
上述代码很简单,利用LinkedList的get(int index)方法,遍历出所有的元素。
但是需要注意的是,get(int index)方法每次都要遍历该索引之前的所有元素,如代码中的一个LinkedList集合,放入了A,B,C,D四个元素。总共需要四次遍历:
第一次遍历打印A:只需遍历一次。
第二次遍历打印B:需要先找到A,然后再找到B打印。
第三次遍历打印C:需要先找到A,然后找到B,最后找到C打印。
第四次遍历打印D:需要先找到A,然后找到B和C,最后找到D打印。
这样如果集合元素很多,越查找到后面(当然此处的get方法进行了优化,查找前半部分从前面开始遍历,查找后半部分从后面开始遍历,但是需要的时间还是很多)花费的时间越多。那么如何改进呢?
2. 迭代器
data:image/s3,"s3://crabby-images/dc7e0/dc7e0b7597e78b005d3c75b65f7793b14034cdd3" alt=""
在LinkedList集合中也有一个内部类ListItr,方法实现大体相似,通过移动游标指向每一次要遍历的元素,不用在遍历某个元素之前都从头开始。其方法实现也比较简单:
data:image/s3,"s3://crabby-images/6f436/6f436de9700dcf57371b492a6b25382259d4404f" alt=""
data:image/s3,"s3://crabby-images/c2722/c2722d07a12403dee259be4d30acf3f8decc15ca" alt=""
data:image/s3,"s3://crabby-images/c4388/c438835428c98f54546dbd1573b230d987f8f75c" alt=""
这里需要重点注意的是modCount字段,前面在增加和删除元素的时候,都会进行自增操作modCount,这是因为如果一边迭代,一边用集合自带的方法进行删除或者新增操作时,都会抛出异常。(使用迭代器的增删方法不会抛异常)。
2.3.9 迭代器和for循环效率差异
data:image/s3,"s3://crabby-images/b5462/b5462b729ddeaa78b0baf19fc0146aa3483c83c1" alt=""
data:image/s3,"s3://crabby-images/54258/5425839baa5e65baaa7953a7302a2a291c07d6a9" alt=""
打印结果为:
data:image/s3,"s3://crabby-images/9aee9/9aee995a9ed5a118c61f964e031eaa835cf902e1" alt=""
10000个元素两者之间都相差一倍多的时间,如果是十万或百万个元素,那么两者之间相差的速度会越来越大。下面通过图形来解释,如图2-8所示。
普通for循环:每次遍历一个索引的元素之前,都要访问之间所有的索引。
data:image/s3,"s3://crabby-images/94ecb/94ecb24f5f637d5674e3166b54129647bdd3b6de" alt=""
●图2-8 LinkedList普通for循环遍历元素
迭代器:每次访问一个元素后,都会用游标记录当前访问元素的位置,遍历一个元素,记录一个位置。如图2-9所示。
data:image/s3,"s3://crabby-images/b9841/b984188386cd5b6446e55e2dd73e9fd381aa5e0c" alt=""
●图2-9 LinkedList迭代器遍历元素