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. 最适合入门的Laravel中级教程(三)表单验证

    做开发有个原则是永远不能信任用户输入的数据: 即便前端已经做了验证: 在后端 php 也必须要再次验证: laravel 为表单验证提供了强大且简单的方案: 创建示例路由: routes/web.ph ...

  2. clone data

    .clone( ) <div class="demo"></div> <script src = "./jquery.js"> ...

  3. 微信小程序---setData

    data:{ obj:{ name:'hello' } } 对data中obj的name字段进行重新赋值. onLoad: function (option) { var value = 'obj.n ...

  4. python入门(四):字符串、编码、random

    1.字符串 字符串基本有两种类型,str和bytes >>> s="a" >>> type(s) <class 'str'>     ...

  5. linux 下将tomcat注册成服务并开机启动

    一.将startup.sh和shutdown.sh新建软连接到/usr/bin ln -s /usr/local/apache-tomcat-8.5.38/bin/startup.sh /usr/bi ...

  6. JVM参数类型

    java -version看版本号(混合模式) java -Xint -version  解释执行 java -Xcomp -version 编译执行 XX参数是不稳定的用来JVM调优和DeBug B ...

  7. Python面向对象中的classmethod类方法和__getattr__方法介绍

    一.classmethod介绍 介绍:@classmethod修饰符我们从名称就可以知道,这是一个类方法,那么和普通的类中的方法有什么不同的 a.类方法,是由类本身调用的,无需实例化类,直接用类本身调 ...

  8. EF ++属性会更新实体

    var lastBaby = await _babyRepository.FirstOrDefaultAsync(); ++lastBaby.sort; -- sort原本为1 -- 最终会生成一条语 ...

  9. iOS开发第三方库一 IQKeyboardManager

    每一个iOS应用的开发者在工作中都会遇到需要用户键盘输入数据的需求,而输入框(UITextField/UITextView)的父界面可能是普通的UIView,也可能是UIScrollView,UITa ...

  10. Some Websites To Learning Laravel

    https://learninglaravel.net/ https://learnku.com/laravel