前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >the one 1.6版本下载

the one 1.6版本下载

作者头像
用户4585225
修改2020-12-18 17:36:40
4240
修改2020-12-18 17:36:40
举报
文章被收录于专栏:用户4585225的专栏

基于java平台,需要jdk环境配置,用于仿真DTN网络,有完整的DTN网络模块,可自由编辑

文件:n459.com/file/25127180-476408662

以下内容无关:

-------------------------------------------分割线---------------------------------------------

从面试角度分析LinkedList源码 注:本系列文章中用到的jdk版本均为java8

LinkedList类图如下:

在这里插入图片描述
在这里插入图片描述

LinkedList底层是由双向链表实现的。链表好比火车,每节车厢包含了车厢和连接下一节车厢的连接点。而双向链表的每个节点不仅有指向下一个节点的指针,还有指向上一个节点的指针。 在LinkedList源码中有一个Node静态类,源码如下:

private static class Node { E item; Node next; Node prev;

代码语言:javascript
复制
Node(Node<E> prev, E element, Node<E> next) {
    this.item = element;
    this.next = next;
    this.prev = prev;
}
12345

} 一个Node节点包含三个部分,分别是

item:数据 next:下一个节点的指针 prev:上一个节点的指针 LinkedList的主要变量如下:

// 集合中的元素数量 transient int size = 0;

/**

  • 首节点的指针.
  • Invariant: (first == null && last == null) ||
  • (first.prev == null && first.item != null) 1

*/ transient Node first;

/**

  • 尾结点的指针.
  • Invariant: (first == null && last == null) ||
  • (last.next == null && last.item != null) 1

*/ transient Node last; 一 添加元素 LinkedList支持想任意节点位置添加元素,不仅提供了集合常用的add()方法,还提供了addFirst()和addLast(),add()方法默认调用addLast()方法,也就是默认是往链表尾部插入元素的。

add()方法源码:

public boolean add(E e) { linkLast(e); return true; } 1.1 尾部插入元素 linkLast()源码如下:

void linkLast(E e) { final Node l = last; final Node newNode = new Node<>(l, e, null); last = newNode; if (l == null) first = newNode; else l.next = newNode; size++; modCount++; } 我们来画张图演示一下如何给链表尾部插入元素:

假如链表中没有元素

对应源码中的if语句,如果没有元素则新增的这个节点为链表中唯一的一个元素,既是首节点,又是尾结点,前一个元素的指针和后一个元素的指针都是null。这里注意head节点不是第一个节点,head节点只是标识了这个链表的地址。

在这里插入图片描述
在这里插入图片描述

假如链表中有元素

对应源码中else语句。先将新增的元素当成Last节点,然后将原来的Last节点的next指向新节点。

else l.next = newNode;

在这里插入图片描述
在这里插入图片描述

一图胜前言,画个图是不是什么都明白了。

1.2 头部插入元素 linkFirst()源码如下:

private void linkFirst(E e) { final Node f = first; final Node newNode = new Node<>(null, e, f); first = newNode; if (f == null) last = newNode; else f.prev = newNode; size++; modCount++; } 还是根据上面的图来解读一下源码,先将第一个节点赋值给中间变量f,将新节点newNode赋值给first节点。如果链表没有元素,则Last节点和First节点都是新插入的节点newNode,否则,将原来的First节点的头指针指向新节点。

二 删除元素 LinkedList提供的删除方法有根据索引和元素删除,除此之外还提供删除第一个元素和最后一个元素的方法,这里我们只分析一下根据索引删除的方法。

public E remove(int index) { checkElementIndex(index); return unlink(node(index)); } checkElementIndex(index)方法就是用来判断传输的索引值是否合法,不合法则抛出数组越界异常。重点来看一下unlink(node(index))方法是如何删除元素的。

node(index)方法源码:

node(index)方法就是根据索引获取该索引位置的节点

Node node(int index) { // assert isElementIndex(index); // 如果指定下标 < 一半元素数量,则从首结点开始遍历 // 否则,从尾结点开始遍历 if (index < (size >> 1)) { Node x = first; for (int i = 0; i < index; i++) x = x.next; return x; } else { Node x = last; for (int i = size - 1; i > index; i–) x = x.prev; return x; } } unlink(Node x)源码如下:

E unlink(Node x) { // assert x != null; final E element = x.item; final Node next = x.next; final Node prev = x.prev;

代码语言:javascript
复制
if (prev == null) {
    first = next;
} else {
    prev.next = next;
    x.prev = null;
}

if (next == null) {
    last = prev;
} else {
    next.prev = prev;
    x.next = null;
}

x.item = null;
size--;
modCount++;
return element;
123456789101112131415161718

} 画张图分析一下删除是如何进行的:

在这里插入图片描述
在这里插入图片描述

假设删除的是第一个元素:则它的prevNULL,我们需要将他的后一个元素(图中的second)作为第一个元素 假设删除的是最后一个元素,则它的nextnull,我们需要将他的前一个元素(途中的second)作为最后一个元素 如果是中间的任意元素,则需要将它的前一个元素的next指针指向它的后一个元素,同时将它的后一个元素的prev指针指向它的前一个元素。 三 总结 LinkedList底层是由双向链表实现的,由于是链表实现的,不仅要存放数据,还要存放指针,所以内存开销要比ArrayList大,删除元素不需要移动其他元素,只需要改变指针的指向,因此删除效率更高,同时它没有实现RandomAccess接口,因此使用迭代器遍历要比for循环更加高效。LinkedList也支持插入重复值和空值,同样也是线程不安全的。

点关注、不迷路

如果觉得文章不错,欢迎关注、点赞、收藏,你们的支持是我创作的动力,感谢大家。

如果文章写的有问题,请不要吝惜文笔,欢迎留言指出,我会及时核查修改。

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文系转载前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档