java linkedlist类_JDK1.8源码(六)——java.util.LinkedList 类

上一篇博客我们介绍了List集合的一种典型实现 ArrayList,我们知道 ArrayList 是由数组构成的,本篇博客我们介绍 List 集合的另一种典型实现 LinkedList,这是一个由链表构成的数组,关于链表的介绍,在这篇博客中 我们也详细介绍过,本篇博客我们将介绍 LinkedList 是如何实现的。

1、LinkedList 定义

LinkedList 是一个用链表实现的集合,元素有序且可以重复。

1 public class LinkedList

2 extends AbstractSequentialList

3 implements List, Deque, Cloneable, java.io.Serializable

e8f0e37d53a0363b44b38cbc9293c0b4.png

和 ArrayList 集合一样,LinkedList 集合也实现了Cloneable接口和Serializable接口,分别用来支持克隆以及支持序列化。List 接口也不用多说,定义了一套 List 集合类型的方法规范。

注意,相对于 ArrayList 集合,LinkedList 集合多实现了一个 Deque 接口,这是一个双向队列接口,双向队列就是两端都可以进行增加和删除操作。

2、字段属性

//链表元素(节点)的个数

transient int size = 0;/***指向第一个节点的指针*/

transient Nodefirst;/***指向最后一个节点的指针*/

transient Node last;

注意这里出现了一个 Node 类,这是 LinkedList 类中的一个内部类,其中每一个元素就代表一个 Node 类对象,LinkedList 集合就是由许多个 Node 对象类似于手拉着手构成。

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 private static class Node{2 E item;//实际存储的元素

3 Node next;//指向上一个节点的引用

4 Node prev;//指向下一个节点的引用5

6 //构造函数

7 Node(Node prev, E element, Nodenext) {8 this.item =element;9 this.next =next;10 this.prev =prev;11 }12 }

View Code

如下图所示:

acb57d1f88a8dbe672ef39c7f2bec5e6.png

上图的 LinkedList 是有四个元素,也就是由 4 个 Node 对象组成,size=4,head 指向第一个elementA,tail指向最后一个节点elementD。

3、构造函数

publicLinkedList() {

}public LinkedList(Collection extends E>c) {this();

addAll(c);

}

LinkedList 有两个构造函数,第一个是默认的空的构造函数,第二个是将已有元素的集合Collection 的实例添加到 LinkedList 中,调用的是 addAll() 方法,这个方法下面我们会介绍。

注意:LinkedList 是没有初始化链表大小的构造函数,因为链表不像数组,一个定义好的数组是必须要有确定的大小,然后去分配内存空间,而链表不一样,它没有确定的大小,通过指针的移动来指向下一个内存地址的分配。

4、添加元素

①、addFirst(E e)

将指定元素添加到链表头

f39cc21603844ae22e818456779af2fe.png

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 //将指定的元素附加到链表头节点

2 public voidaddFirst(E e) {3 linkFirst(e);4 }5 private voidlinkFirst(E e) {6 final Node f = first;//将头节点赋值给 f

7 final Node newNode = new Node<>(null, e, f);//将指定元素构造成一个新节点,此节点的指向下一个节点的引用为头节点

8 first = newNode;//将新节点设为头节点,那么原先的头节点 f 变为第二个节点

9 if (f == null)//如果第二个节点为空,也就是原先链表是空

10 last = newNode;//将这个新节点也设为尾节点(前面已经设为头节点了)

11 else

12 f.prev = newNode;//将原先的头节点的上一个节点指向新节点

13 size++;//节点数加1

14 modCount++;//和ArrayList中一样,iterator和listIterator方法返回的迭代器和列表迭代器实现使用。

15 }

View Code

②、addLast(E e)和add(E e)

将指定元素添加到链表尾

d86e13fe7bc483d1a96a94b500cf4f2a.png

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 //将元素添加到链表末尾

2 public voidaddLast(E e) {3 linkLast(e);4 }5 //将元素添加到链表末尾

6 public booleanadd(E e) {7 linkLast(e);8 return true;9 }10 voidlinkLast(E e) {11 final Node l = last;//将l设为尾节点

12 final Node newNode = new Node<>(l, e, null);//构造一个新节点,节点上一个节点引用指向尾节点l

13 last = newNode;//将尾节点设为创建的新节点

14 if (l == null)//如果尾节点为空,表示原先链表为空

15 first = newNode;//将头节点设为新创建的节点(尾节点也是新创建的节点)

16 else

17 l.next = newNode;//将原来尾节点下一个节点的引用指向新节点

18 size++;//节点数加1

19 modCount++;//和ArrayList中一样,iterator和listIterator方法返回的迭代器和列表迭代器实现使用。

20 }

View Code

③、add(int index, E element)

将指定的元素插入此列表中的指定位置

dd7e0275a7fb059eda900f9f9f9c2eab.png

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

//将指定的元素插入此列表中的指定位置

public void add(intindex, E element) {//判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常

checkPositionIndex(index);if (index == size)//如果索引值等于链表大小

linkLast(element);//将节点插入到尾节点

elselinkBefore(element, node(index));

}voidlinkLast(E e) {final Node l = last;//将l设为尾节点

final Node newNode = new Node<>(l, e, null);//构造一个新节点,节点上一个节点引用指向尾节点l

last = newNode;//将尾节点设为创建的新节点

if (l == null)//如果尾节点为空,表示原先链表为空

first = newNode;//将头节点设为新创建的节点(尾节点也是新创建的节点)

elsel.next= newNode;//将原来尾节点下一个节点的引用指向新节点

size++;//节点数加1

modCount++;//和ArrayList中一样,iterator和listIterator方法返回的迭代器和列表迭代器实现使用。

}

Node node(intindex) {if (index < (size >> 1)) {//如果插入的索引在前半部分

Node x = first;//设x为头节点

for (int i = 0; i < index; i++)//从开始节点到插入节点索引之间的所有节点向后移动一位

x =x.next;returnx;

}else {//如果插入节点位置在后半部分

Node x = last;//将x设为最后一个节点

for (int i = size - 1; i > index; i--)//从最后节点到插入节点的索引位置之间的所有节点向前移动一位

x =x.prev;returnx;

}

}void linkBefore(E e, Nodesucc) {final Node pred = succ.prev;//将pred设为插入节点的上一个节点

final Node newNode = new Node<>(pred, e, succ);//将新节点的上引用设为pred,下引用设为succ

succ.prev = newNode;//succ的上一个节点的引用设为新节点

if (pred == null)//如果插入节点的上一个节点引用为空

first = newNode;//新节点就是头节点

elsepred.next= newNode;//插入节点的下一个节点引用设为新节点

size++;

modCount++;

}

View Code

④、addAll(Collection extends E> c)

按照指定集合的​​迭代器返回的顺序,将指定集合中的所有元素追加到此列表的末尾

此方法还有一个 addAll(int index, Collection extends E> c),将集合 c 中所有元素插入到指定索引的位置。其实

addAll(Collection extends E> c) ==  addAll(size, Collection extends E> c)

源码如下:

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 //按照指定集合的​​迭代器返回的顺序,将指定集合中的所有元素追加到此列表的末尾。

2 public boolean addAll(Collection extends E>c) {3 returnaddAll(size, c);4 }5 //将集合 c 中所有元素插入到指定索引的位置。

6 public boolean addAll(int index, Collection extends E>c) {7 //判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常

8 checkPositionIndex(index);9

10 Object[] a = c.toArray();//将集合转换成一个 Object 类型的数组

11 int numNew =a.length;12 if (numNew == 0)//如果添加的集合为空,直接返回false

13 return false;14

15 Nodepred, succ;16 if (index == size) {//如果插入的位置等于链表的长度,就是将原集合元素附加到链表的末尾

17 succ = null;18 pred =last;19 } else{20 succ =node(index);21 pred =succ.prev;22 }23

24 for (Object o : a) {//遍历要插入的元素

25 @SuppressWarnings("unchecked") E e =(E) o;26 Node newNode = new Node<>(pred, e, null);27 if (pred == null)28 first =newNode;29 else

30 pred.next =newNode;31 pred =newNode;32 }33

34 if (succ == null) {35 last =pred;36 } else{37 pred.next =succ;38 succ.prev =pred;39 }40

41 size +=numNew;42 modCount++;43 return true;44 }

View Code

看到上面向 LinkedList 集合中添加元素的各种方式,我们发现LinkedList 每次添加元素只是改变元素的上一个指针引用和下一个指针引用,而且没有扩容。,对比于 ArrayList ,需要扩容,而且在中间插入元素时,后面的所有元素都要移动一位,两者插入元素时的效率差异很大,下一篇博客会对这两者的效率,以及何种情况选择何种集合进行分析。

还有,每次进行添加操作,都有modCount++ 的操作,

5、删除元素

删除元素和添加元素一样,也是通过更改指向上一个节点和指向下一个节点的引用即可,这里就不作图形展示了。

①、remove()和removeFirst()

从此列表中移除并返回第一个元素

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 //从此列表中移除并返回第一个元素

2 publicE remove() {3 returnremoveFirst();4 }5 //从此列表中移除并返回第一个元素

6 publicE removeFirst() {7 final Node f = first;//f设为头结点

8 if (f == null)9 throw new NoSuchElementException();//如果头结点为空,则抛出异常

10 returnunlinkFirst(f);11 }12 private E unlinkFirst(Nodef) {13 //assert f == first && f != null;

14 final E element =f.item;15 final Node next = f.next;//next 为头结点的下一个节点

16 f.item = null;17 f.next = null; //将节点的元素以及引用都设为 null,便于垃圾回收

18 first = next; //修改头结点为第二个节点

19 if (next == null)//如果第二个节点为空(当前链表只存在第一个元素)

20 last = null;//那么尾节点也置为 null

21 else

22 next.prev = null;//如果第二个节点不为空,那么将第二个节点的上一个引用置为 null

23 size--;24 modCount++;25 returnelement;26 }

View Code

②、removeLast()

从该列表中删除并返回最后一个元素

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 //从该列表中删除并返回最后一个元素

2 publicE removeLast() {3 final Node l =last;4 if (l == null)//如果尾节点为空,表示当前集合为空,抛出异常

5 throw newNoSuchElementException();6 returnunlinkLast(l);7 }8

9 private E unlinkLast(Nodel) {10 //assert l == last && l != null;

11 final E element =l.item;12 final Node prev =l.prev;13 l.item = null;14 l.prev = null; //将节点的元素以及引用都设为 null,便于垃圾回收

15 last = prev;//尾节点为倒数第二个节点

16 if (prev == null)//如果倒数第二个节点为null

17 first = null;//那么将节点也置为 null

18 else

19 prev.next = null;//如果倒数第二个节点不为空,那么将倒数第二个节点的下一个引用置为 null

20 size--;21 modCount++;22 returnelement;23 }

View Code

③、remove(int index)

删除此列表中指定位置的元素

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 //删除此列表中指定位置的元素

2 public E remove(intindex) {3 //判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常

4 checkElementIndex(index);5 returnunlink(node(index));6 }7 E unlink(Nodex) {8 //assert x != null;

9 final E element =x.item;10 final Node next =x.next;11 final Node prev =x.prev;12

13 if (prev == null) {//如果删除节点位置的上一个节点引用为null(表示删除第一个元素)

14 first = next;//将头结点置为第一个元素的下一个节点

15 } else {//如果删除节点位置的上一个节点引用不为null

16 prev.next = next;//将删除节点的上一个节点的下一个节点引用指向删除节点的下一个节点(去掉删除节点)

17 x.prev = null;//删除节点的上一个节点引用置为null

18 }19

20 if (next == null) {//如果删除节点的下一个节点引用为null(表示删除最后一个节点)

21 last = prev;//将尾节点置为删除节点的上一个节点

22 } else {//不是删除尾节点

23 next.prev = prev;//将删除节点的下一个节点的上一个节点的引用指向删除节点的上一个节点

24 x.next = null;//将删除节点的下一个节点引用置为null

25 }26

27 x.item = null;//删除节点内容置为null,便于垃圾回收

28 size--;29 modCount++;30 returnelement;31 }

View Code

④、remove(Object o)

如果存在,则从该列表中删除指定元素的第一次出现

此方法本质上和 remove(int index) 没多大区别,通过循环判断元素进行删除,需要注意的是,是删除第一次出现的元素,不是所有的。

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 public booleanremove(Object o) {2 if (o == null) {3 for (Node x = first; x != null; x =x.next) {4 if (x.item == null) {5 unlink(x);6 return true;7 }8 }9 } else{10 for (Node x = first; x != null; x =x.next) {11 if(o.equals(x.item)) {12 unlink(x);13 return true;14 }15 }16 }17 return false;18 }

View Code

6、修改元素

通过调用 set(int index, E element) 方法,用指定的元素替换此列表中指定位置的元素。

public E set(intindex, E element) {//判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常

checkElementIndex(index);

Node x = node(index);//获取指定索引处的元素

E oldVal =x.item;

x.item= element;//将指定位置的元素替换成要修改的元素

return oldVal;//返回指定索引位置原来的元素

}

这里主要是通过 node(index) 方法获取指定索引位置的节点,然后修改此节点位置的元素即可。

7、查找元素

①、getFirst()

返回此列表中的第一个元素

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 publicE getFirst() {2 final Node f =first;3 if (f == null)4 throw newNoSuchElementException();5 returnf.item;6 }

View Code

②、getLast()

返回此列表中的最后一个元素

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 publicE getLast() {2 final Node l =last;3 if (l == null)4 throw newNoSuchElementException();5 returnl.item;6 }

View Code

③、get(int index)

返回指定索引处的元素

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 public E get(intindex) {2 checkElementIndex(index);3 returnnode(index).item;4 }

View Code

④、indexOf(Object o)

返回此列表中指定元素第一次出现的索引,如果此列表不包含元素,则返回-1。

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 //返回此列表中指定元素第一次出现的索引,如果此列表不包含元素,则返回-1。

2 public intindexOf(Object o) {3 int index = 0;4 if (o == null) {//如果查找的元素为null(LinkedList可以允许null值)

5 for (Node x = first; x != null; x = x.next) {//从头结点开始不断向下一个节点进行遍历

6 if (x.item == null)7 returnindex;8 index++;9 }10 } else {//如果查找的元素不为null

11 for (Node x = first; x != null; x =x.next) {12 if(o.equals(x.item))13 returnindex;14 index++;15 }16 }17 return -1;//找不到返回-1

18 }

View Code

8、遍历集合

①、普通 for 循环

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

LinkedList linkedList = new LinkedList<>();

linkedList.add("A");

linkedList.add("B");

linkedList.add("C");

linkedList.add("D");for(int i = 0 ; i < linkedList.size() ; i++){

System.out.print(linkedList.get(i)+" ");//A B C D

}

View Code

代码很简单,我们就利用 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方法进行了优化,查找前半部分从前面开始遍历,查找后半部分从后面开始遍历,但是需要的时间还是很多)花费的时间越多。那么如何改进呢?

②、迭代器

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 LinkedList linkedList = new LinkedList<>();2 linkedList.add("A");3 linkedList.add("B");4 linkedList.add("C");5 linkedList.add("D");6

7

8 Iterator listIt =linkedList.listIterator();9 while(listIt.hasNext()){10 System.out.print(listIt.next()+" ");//A B C D

11 }12

13 //通过适配器模式实现的接口,作用是倒叙打印链表

14 Iterator it =linkedList.descendingIterator();15 while(it.hasNext()){16 System.out.print(it.next()+" ");//D C B A

17 }

View Code

在 LinkedList 集合中也有一个内部类 ListItr,方法实现大体上也差不多,通过移动游标指向每一次要遍历的元素,不用在遍历某个元素之前都要从头开始。其方法实现也比较简单:

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 public ListIterator listIterator(intindex) {2 checkPositionIndex(index);3 return newListItr(index);4 }5

6 private class ListItr implements ListIterator{7 private NodelastReturned;8 private Nodenext;9 private intnextIndex;10 private int expectedModCount =modCount;11

12 ListItr(intindex) {13 //assert isPositionIndex(index);

14 next = (index == size) ? null: node(index);15 nextIndex =index;16 }17

18 public booleanhasNext() {19 return nextIndex

22 publicE next() {23 checkForComodification();24 if (!hasNext())25 throw newNoSuchElementException();26

27 lastReturned =next;28 next =next.next;29 nextIndex++;30 returnlastReturned.item;31 }32

33 public booleanhasPrevious() {34 return nextIndex > 0;35 }36

37 publicE previous() {38 checkForComodification();39 if (!hasPrevious())40 throw newNoSuchElementException();41

42 lastReturned = next = (next == null) ?last : next.prev;43 nextIndex--;44 returnlastReturned.item;45 }46

47 public intnextIndex() {48 returnnextIndex;49 }50

51 public intpreviousIndex() {52 return nextIndex - 1;53 }54

55 public voidremove() {56 checkForComodification();57 if (lastReturned == null)58 throw newIllegalStateException();59

60 Node lastNext =lastReturned.next;61 unlink(lastReturned);62 if (next ==lastReturned)63 next =lastNext;64 else

65 nextIndex--;66 lastReturned = null;67 expectedModCount++;68 }69

70 public voidset(E e) {71 if (lastReturned == null)72 throw newIllegalStateException();73 checkForComodification();74 lastReturned.item =e;75 }76

77 public voidadd(E e) {78 checkForComodification();79 lastReturned = null;80 if (next == null)81 linkLast(e);82 else

83 linkBefore(e, next);84 nextIndex++;85 expectedModCount++;86 }87

88 public void forEachRemaining(Consumer super E>action) {89 Objects.requireNonNull(action);90 while (modCount == expectedModCount && nextIndex

99 final voidcheckForComodification() {100 if (modCount !=expectedModCount)101 throw newConcurrentModificationException();102 }103 }

View Code

这里需要重点注意的是 modCount 字段,前面我们在增加和删除元素的时候,都会进行自增操作 modCount,这是因为如果想一边迭代,一边用集合自带的方法进行删除或者新增操作,都会抛出异常。(使用迭代器的增删方法不会抛异常)

final voidcheckForComodification() {if (modCount !=expectedModCount)throw newConcurrentModificationException();

}

比如:

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

1 LinkedList linkedList = new LinkedList<>();2 linkedList.add("A");3 linkedList.add("B");4 linkedList.add("C");5 linkedList.add("D");6

7

8 Iterator listIt =linkedList.listIterator();9 while(listIt.hasNext()){10 System.out.print(listIt.next()+" ");//A B C D11 //linkedList.remove();//此处会抛出异常

12 listIt.remove();//这样可以进行删除操作

13 }

View Code

迭代器的另一种形式就是使用 foreach 循环,底层实现也是使用的迭代器,这里我们就不做介绍了。

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

LinkedList linkedList = new LinkedList<>();

linkedList.add("A");

linkedList.add("B");

linkedList.add("C");

linkedList.add("D");for(String str : linkedList){

System.out.print(str+ "");

}

View Code

9、迭代器和for循环效率差异

1 LinkedList linkedList = new LinkedList<>();2 for(int i = 0 ; i < 10000 ; i++){//向链表中添加一万个元素

3 linkedList.add(i);4 }5 long beginTimeFor =System.currentTimeMillis();6 for(int i = 0 ; i < 10000 ; i++){7 System.out.print(linkedList.get(i));8 }9 long endTimeFor =System.currentTimeMillis();10 System.out.println("使用普通for循环遍历10000个元素需要的时间:"+ (endTimeFor -beginTimeFor));11

12

13 long beginTimeIte =System.currentTimeMillis();14 Iterator it =linkedList.listIterator();15 while(it.hasNext()){16 System.out.print(it.next()+" ");17 }18 long endTimeIte =System.currentTimeMillis();19 System.out.println("使用迭代器遍历10000个元素需要的时间:"+ (endTimeIte - beginTimeIte));

打印结果为:

e51d9d03bfbf2f8337a8670b34232753.png

一万个元素两者之间都相差一倍多的时间,如果是十万,百万个元素,那么两者之间相差的速度会越来越大。下面通过图形来解释:

普通for循环:每次遍历一个索引的元素之前,都要访问之间所有的索引。

ff234ffc2a96329895e86226a86b47b9.png

迭代器:每次访问一个元素后,都会用游标记录当前访问元素的位置,遍历一个元素,记录一个位置。

23f309f419e269bea64e5fd46bd0969e.png

参考文档:https://docs.oracle.com/javase/8/docs/api/java/util/LinkedList.html#

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值