jdk版本:1.8

LinkedList添加元素有两个方法:add(E e)和add(int index,E e)。

add(E e)

/**
* Appends the specified element to the end of this list.
* 在列表最后添加指定元素
*/
public boolean add(E e) {
linkLast(e);
return true;
}

add(E e)是直接在队尾添加元素。再看一下linkLast(E e)方法,源码如下。

void linkLast(E e) {
//找到链表的最后一个节点,赋值给l,
final Node<E> l = last;
//创建节点,这个节点的上一个节点就是上面last节点
final Node<E> newNode = new Node<>(l, e, null);
//将新的节点赋值给最后节点
last = newNode;
if (l == null)
//如果没有最后节点,表示添加链表为空,赋值给首节点。
first = newNode;
else
//有最后一个节点,将最后节点的next指针指向新建的节点
l.next = newNode;
size++;
modCount++;
}
  1. LinkedList会记录链表的最后一个节点last,
  2. 首先创建新的节点,新节点的pre就是队列的最后一个节点last,新节点的next为null,
  3. 如果last为空表示这个链表为空,新节点就是首节点first。
  4. 如果last不为空表示链表不为空,将last节点的next指针指向新节点。

add(int index,E e)

add(int index,E e)是根据元素插入到指定位置上,index表示链表的位置

/**
* Inserts the specified element at the specified position in this list.
* 插入指定的元素到列表指定位置上
*/
public void add(int index, E element) {
//检查位置是否越界
checkPositionIndex(index);
//如果插入的下标等于链表的大小,直接就是添加到队尾,
if (index == size)
linkLast(element);
else
linkBefore(element, node(index));
}
  1. 首先检查index是否超过链表大小,即index 是否会大于链表的size。
  2. 如果index等于链表的大小,添加元素就是在链表队尾添加元素,和add(E e) 操作一致。
  3. 如果不等会size大小就调用linkBefore方法,首先在该方法的第二个参数使用了node方法。node方法源码如下:
 Node<E> node(int index) {
//size >>1 表示size右移,表示size的一半
//如果index小于size一半,从首节点往后遍历
if (index < (size >> 1)) {
Node<E> x = first;
for (int i = 0; i < index; i++)
x = x.next;
return x;
//如果index大于size一半,从最后一个节点往前遍历
} else {
Node<E> x = last;
for (int i = size - 1; i > index; i--)
x = x.prev;
return x;
}
}

node方法通过链表的位置找到链表的元素。这里用到了一个size的右移运算,size>>1表示size/2。首先判断index是在链表的前一半还是后一半,因为linkedList是双链表,可以往前和往后进行遍历。如果在前半部分就从首节点往后遍历,如果在后半部分就从最后一个节点往前遍历,这样最多遍历size的一半,避免遍历整个链表。

找到index对应的元素后执行linkedBefore方法

/**
* Inserts element e before non-null Node succ.
* 往succ节点前面插入元素
*/
void linkBefore(E e, Node<E> succ) {
//succ的前一个节点
final Node<E> pred = succ.prev;
//创建新节点,新节点的pre就是succ的pre节点,新节点的next就是succ节点
final Node<E> newNode = new Node<>(pred, e, succ);
//succ.pre指向新节点
succ.prev = newNode;
//如果succ前节点为空,表示succ就是首节点,新节点即为首节点
if (pred == null)
first = newNode;
else
//succ的上一节点的next指向新节点
pred.next = newNode;
size++;
modCount++;
}
  1. 新建节点,节点pre就是succ的pre,节点的next就是succ。
  2. 将succ.pre指向新节点。
  3. succ的pre为null,succ即为首节点,将first赋值给首节点
  4. succ的pre不为空,则把succ的上一节点的next指向新节点

总结

  1. LinkedList,只有两种添加方式,一种是在列表最后添加(linkLast),一种是在列表某个元素前面添加 (linkBefore)。
  2. LinkLast,首先创建一个新节点,节点pre指向最后一个节点,最后一个节点的next指向新节点。
  3. LinkBefore,首先根据index下标获取到元素的位置,新建新节点,新节点的pre就是元素的pre,新节点的next就是该元素。元素的pre的next指向新元素。

为何Linked要用双链表而不是单链表

LinkedList为何是双链表,链表主要缺点是查询速度很慢,添加或者删除都要找到要添加和删除的节点,而使用双链表,每次遍历循环前,都会判断一下索引是在链表的前半部分还是后半部分。如果是前半部分的话,从首部遍历到中间。如果是后半部分,从尾部遍历到中间。

LinkedList 添加元素源码解析的更多相关文章

  1. Java集合---LinkedList源码解析

    一.源码解析1. LinkedList类定义2.LinkedList数据结构原理3.私有属性4.构造方法5.元素添加add()及原理6.删除数据remove()7.数据获取get()8.数据复制clo ...

  2. 给jdk写注释系列之jdk1.6容器(2)-LinkedList源码解析

    LinkedList是基于链表结构的一种List,在分析LinkedList源码前有必要对链表结构进行说明.   1.链表的概念      链表是由一系列非连续的节点组成的存储结构,简单分下类的话,链 ...

  3. LinkedList源码解析

    LinkedList是基于链表结构的一种List,在分析LinkedList源码前有必要对链表结构进行说明.1.链表的概念链表是由一系列非连续的节点组成的存储结构,简单分下类的话,链表又分为单向链表和 ...

  4. 从源码解析LinkedList集合

         上篇文章我们介绍了ArrayList类的基本的使用及其内部的一些方法的实现原理,但是这种集合类型虽然可以随机访问数据,但是如果需要删除中间的元素就需要移动一半的元素的位置,效率低下.并且它内 ...

  5. ava集合---LinkedList源码解析

    一.源码解析 public class LinkedList<E> extends AbstractSequentialList<E> implements List<E ...

  6. Java集合:LinkedList源码解析

    Java集合---LinkedList源码解析   一.源码解析1. LinkedList类定义2.LinkedList数据结构原理3.私有属性4.构造方法5.元素添加add()及原理6.删除数据re ...

  7. Java 集合系列05之 LinkedList详细介绍(源码解析)和使用示例

    概要  前面,我们已经学习了ArrayList,并了解了fail-fast机制.这一章我们接着学习List的实现类——LinkedList.和学习ArrayList一样,接下来呢,我们先对Linked ...

  8. Java 集合系列 04 LinkedList详细介绍(源码解析)和使用示例

    java 集合系列目录: Java 集合系列 01 总体框架 Java 集合系列 02 Collection架构 Java 集合系列 03 ArrayList详细介绍(源码解析)和使用示例 Java ...

  9. 有关LinkedList常用方法的源码解析

    上文里解析了有关ArrayList中的几个常用方法的源码——<有关ArrayList常用方法的源码解析>,本文将对LinkedList的常用方法做简要解析. LinkedList是基于链表 ...

随机推荐

  1. 模拟input type=file

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  2. 第1篇-关于JVM运行时,开篇说的简单些

    开讲Java运行时,这一篇讲一些简单的内容.我们写的主类中的main()方法是如何被Java虚拟机调用到的?在Java类中的一些方法会被由C/C++编写的HotSpot虚拟机的C/C++函数调用,不过 ...

  3. 阿里面试官:Android中binder机制的实现原理及过程?

    Binder 是 Android 系统中非常重要的组成部分.Android 系统中的许多功能建立在 Binder 机制之上.在这篇文章中,我们会对 Android 中的 Binder 在系统架构中的作 ...

  4. 树莓派SG90舵机接法

    我的舵机的三条线是红的.黑色.棕色,接法如下: 棕 : GND 红 : VCC 黄: 信号线 如图所示: 图片来源 如上图所示,写代码时注意舵机的BCM编码是18,而不是物理引脚的编码12.

  5. git config 配置简写命令

    在多人协作开发时,一般用git来进行代码管理. git有一些命令如:git pull . git push等等,这些命令可以设置alias,也就是缩写. 如:git pull 是 git pl, gi ...

  6. 消息协议AMQP 与 JMS对比

    https://blog.csdn.net/hpttlook/article/details/23391967 https://www.jianshu.com/p/6e6821604efc https ...

  7. Redis实现分布式锁那件事

    今天我们来聊一聊分布式锁的那些事. 相信大家对锁已经不陌生了,我们在多线程环境中,如果需要对同一个资源进行操作,为了避免数据不一致,我们需要在操作共享资源之前进行加锁操作.在计算机科学中,锁(lock ...

  8. ELK太重?试试KFC日志采集

    写在前面 ELK三剑客(ElasticSearch,Logstash,Kibana)基本上可以满足日志采集.信息处理.统计分析.可视化报表等一些日志分析的工作,但是对我们来说--太重了,并且技术栈不是 ...

  9. ACL的配置

    一.实验拓扑 实验要求: 二.实验编址 三.实验步骤: 1.启动设备(全选) 2.配置端口IP R1: R2: R3: R4: 2.搭建OSPF网络: R1: R2: R3: R4: 4.配置ACL控 ...

  10. 在java程序中使用protobuf

    目录 简介 为什么使用protobuf 定义.proto文件 编译协议文件 详解生成的文件 Builders 和 Messages 序列化和反序列化 协议扩展 总结 简介 Protocol Buffe ...