LinkedList是基于双向链表数据结构来存储数据的,以下是对LinkedList  的 属性,构造器 ,add(E e),remove(index),get(Index),set(inde,e)进行源码分析:

属性:

transient int size = 0;    //记录集合的大小

    /**
* Pointer to first node.
* Invariant: (first == null && last == null) ||
* (first.prev == null && first.item != null)
*/
transient Node<E> first; //指向首节点对象 /**
* Pointer to last node.
* Invariant: (first == null && last == null) ||
* (last.next == null && last.item != null)
*/
transient Node<E> last; //指向末节点对象

2构造器:

public LinkedList() {   //构造空的LinkedList对象
}
 public LinkedList(Collection<? extends E> c) {   //构造对象,将集合元素添加到新集合中
this();
addAll(c);
}

3:方法:add(E e)

public boolean add(E e) {
linkLast(e);
return true;
}

linkedLast(e) 源码

/**
* Links e as last element.
*/
void linkLast(E e) {
final Node<E> l = last; //将原来的最末节点对象暂存 l 引用
final Node<E> newNode = new Node<>(l, e, null); /构建新的Node对象
last = newNode; //将链表对象的last引用指向新增的节点元素
if (l == null)
first = newNode; //如果不存在之前指向的节点,则first引用指向新创建的节点对象
else
l.next = newNode; //存在前一个节点,之前最后节点对象的next指向新建的节点对象
size++; //结合的长度加1
modCount++;
}

Node对象的构造器如下:

private static class Node<E> {
E item;
Node<E> next;
Node<E> prev; Node(Node<E> prev, E element, Node<E> next) { //参数为 l:之前的最后一个节点, element:需要新增的元素, next null
this.item = element; //要增加的元素
this.next = next; //新增节点的next指向为null
this.prev = prev; //新增节点的prev指向之前的节点
}
}

remove方法:

 public E remove(int index) {    //删除指定索引的元素
checkElementIndex(index); //检查是否索引越界
return unlink(node(index));
}
node(index) 的源码如下:
Node<E> node(int index) {
// assert isElementIndex(index); if (index < (size >> 1)) { //获取到一般长度的集合索引值
Node<E> x = first; //暂存链表中首节点对象
for (int i = 0; i < index; i++) //遍历前半段集合节点
x = x.next;
return x;
} else {
Node<E> x = last;
for (int i = size - 1; i > index; i--)
x = x.prev;
return x;
}
}

这里有点繁琐,举个具体的实例说明:比如需要删除index=5;的节点对象,假设结合的长度为20

则调用 node(5) 方法后返回的是什么呢?假设Node(0) 为起始位置  

此时:初始:x=Node(0),当i=0   x=Node(1)    i=1   x=Node(2)…… 当i=5-1  x=Node(5)   此时就定位到了需要删除的节点对象 即 Node(index)

接下来调用:   unlink(node(index))  继续以index=5为例

E unlink(Node<E> x) {
// assert x != null;
final E element = x.item; //Node(5).data
final Node<E> next = x.next; //next=Node(6)
final Node<E> prev = x.prev; //prev=Node(4) if (prev == null) {
first = next;
} else {
prev.next = next; //Node(4).next=Node(6)
x.prev = null; //Node(5).prev=null
} if (next == null) {
last = prev;
} else {
next.prev = prev; // Node(6).prev=Node(4)
x.next = null; //Node(5).next=null 回收
} x.item = null; //Node(5)=null
size--;
modCount++;
return element;
}

这样就完成了  Node(index-1).next=Node(index+1)   Node(index+1).prev=Node(index-1)   Node(index).data=null  Node(index).prev=null  Node(index).next=null  完成了删除动作  删除相应的索引的节点

删除第一个节点和删除最后一个节点的原理类似;

Get(int index) 方法:

public E get(int index) {
checkElementIndex(index); //检查索引是否越界
return node(index).item; //node(index) 在删除的方法中分析过,返回索引为index的节点对象, 所以get方法 返回的是该索引节点的存储数据对象
}

set(index,e) 方法:

public E set(int index, E element) {
checkElementIndex(index);
Node<E> x = node(index); //调用node(index)放回Node(index)
E oldVal = x.item;
x.item = element; //将 Node(index)的引用指向新的对象
return oldVal; }

 到此LinkedList的源码分析结束了:

mark:使用LinkedList 时,使用的是链表结构,当调用add()方法时,默认添加到最后一个,集合不需要扩充,减少内存消耗;

但是当LinkedList 进行指定索引的查询,元素替换,删除,需要对集合从first指向开始进行遍历一遍才能进行,有相应的计算复杂度;使用时应当考虑到这一点 

  

  

  

  

  

  

LinkedList 的源码分析的更多相关文章

  1. LinkedList的源码分析

    1. LinkedList的定义  1.1  继承于AbstractSequentialList的双向链表,可以被当作堆栈.队列或双端队列进行操作 1.2  有序,非线程安全的双向链表,默认使用尾部插 ...

  2. Java的LinkedList底层源码分析

    首先我们先说一下,源码里可以看出此类不仅仅用双向链表实现了队列数据结构的功能,还提供了链表数据结构的功能.

  3. LinkedList<E>源码分析

    LinkedList的数据结构就是双向链表,如下所示: private static class Node<E> { E item;//数据元素 Node<E> next;// ...

  4. LinkedList的源码分析(基于jdk1.8)

    1.初始化 public LinkedList() { } 并未开辟任何类似于数组一样的存储空间,那么链表是如何存储元素的呢? 2.Node类型 存储到链表中的元素会被封装为一个Node类型的结点.并 ...

  5. Java——LinkedList底层源码分析

    1.简介 LinkedList 是用链表结构存储数据的,很适合数据的动态插入和删除,随机访问和遍历速度比较慢.另外,他还提供了 List 接口中没有定义的方法,专门用于操作表头和表尾元素,可以当作堆栈 ...

  6. 集合框架——LinkedList集合源码分析

    目录 示例代码 底层代码 第1步(初始化集合) 第2步(往集合中添加一个元素) 第3步(往集合中添加第二个元素) 第4步(往集合中添加第三个元素) LinkedList添加元素流程示意图 第5步(删除 ...

  7. 集合之LinkedList(含JDK1.8源码分析)

    一.前言 LinkedList是基于链表实现的,所以先讲解一下什么是链表.链表原先是C/C++的概念,是一种线性的存储结构,意思是将要存储的数据存在一个存储单元里面,这个存储单元里面除了存放有待存储的 ...

  8. LinkedList源码分析和实例应用

    1. LinkedList介绍 LinkedList是继承于AbstractSequentialList抽象类,它也可以被当作堆栈.队列或者双端队列使用. LinkedList实现了Deque接口,即 ...

  9. [源码分析]ArrayList和LinkedList如何实现的?我看你还有机会!

    文章已经收录在 Github.com/niumoo/JavaNotes ,更有 Java 程序员所需要掌握的核心知识,欢迎Star和指教. 欢迎关注我的公众号,文章每周更新. 前言 说真的,在 Jav ...

随机推荐

  1. [Ting's笔记Day8]活用套件carrierwave gem:(3)Deploy图片上传功能到Heroku网站

    前情提要: 身为Ruby新手村民,创造稳定且持续的学习步调很重要,我用的方法就是一周在IT邦写三篇笔记,希望藉由把笔记和遇到的bug记录下来的过程,能帮助到未来想用Ruby on Rails架站的新手 ...

  2. vue浏览器滚动加载更多

    created () { var that = this; window.addEventListener('scroll',this.scroll,true) console.log(this.$r ...

  3. Lua 和 C++ 交互

    step1.搭建好vs和lua相交互的环境: 1.下载一个lua5.3的源码: 2.有Lua_lib.lib这个文件: 3.开始配置: 鼠标放在工程名那: 右键选属性: 把包含目录点开进行编辑: 地址 ...

  4. Selenium Java关闭浏览器

    在学习selenium的过程中发现一个问题,各种博客/教程都是教人用selenium的quit()和close()方法关闭浏览器. 但这不是我要的结果.这两个方法的前提是,用webdriver打开浏览 ...

  5. oracle 执行顺序 select查询优化

    今天把这几天做的练习复习了一下,不知道自己写得代码执行的效率如何以及要如何提高,于是乎上网开始研究一些材料,现整理如下: 首先,要了解在Oracle中Sql语句运行的机制.以下是sql语句的执行步骤: ...

  6. vue 关于父组件无法触发子组件的事件的解决方法

    一般情况导致无法触发子组件的方法  基本都是由于子组件未渲染完成 就进行了调用,解决方法如下: 1.加定时器  setTimeout(() => { //加定时器原因是,子组件页面未渲染处理就做 ...

  7. map与forEach区别

    1.  forEach()返回的是undefined 不可以链式调用 return没有用 2. map()返回一个新数组 原数组不会改 3. 没办法终止或者跳过forEach()和map循环  除非抛 ...

  8. 集群环境下定时调度的解决方案之Quartz集群

    集群环境可能出现的问题 在上一篇博客我们介绍了如何在自己的项目中从无到有的添加了Quartz定时调度引擎,其实就是一个Quartz 和Spring的整合过程,很容易实现,但是我们现在企业中项目通常都是 ...

  9. C++中重载操作符[ ]

    1.首先说说为什么要重载操作符[ ] 主要是因为系统只给了整数类型(int)的重载函数,即只能在方括号中输入整数进行查找,但有时候我们可能存放数据时,下标的类型是自定义的,希望也能像数组直接通过下标访 ...

  10. 【Mybatis】MyBatis快速入门(一)

    Mybatis简介 MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可 ...