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. BUILDING WITH BOOTSTRAP

    BUILDING WITH BOOTSTRAP Bootstrap Generalizations You just built an impressive webpage using the Boo ...

  2. txt,csv,json互相转化

    也没啥,记下来怕忘了.说明都在代码里面: 麻蛋,这个着色好难看 import csv import json #从txt变为csv student_txt=[]; with open("st ...

  3. iptables随笔

    iptables 分为四表五链 四表: filter表 nat 表 mangle 表 raw 表 五链 INPUT 链 OUTPUT 链 FORWARD 链 PREROUTING(路由前) POSTR ...

  4. Hadoop 学习之Docker

    docker环境的配置请参考:http://www.cnblogs.com/frankliu/p/8052673.html hadoop-docker安装地址参考:https://hub.docker ...

  5. springmvc 怎么响应json数据

    springmvc 怎么响应json数据@Controller@RequestMapping("/items") class ItemsController{  @RequestM ...

  6. servlet中的请求响应与重定向区别

    一.概念 请求响应(转发):将客户端请求转发另一个servlet或者jsp页面------------------------getRequestDispatcher()方法 重定向: 返回一个连接给 ...

  7. jquery抓娃娃机代码

    <html><head><title>jQuery抓娃娃机游戏代码 - 源码之家</title><meta name="content- ...

  8. React createRef:引用

    一 代码 import React, { Component } from 'react'; class Box extends Component { render() { return <b ...

  9. 一个域名下多个Vue项目

    公司写的网站要英文和中文的,所以就写了两个项目,都是用vue写的单页面项目,但是域名只有一个,所以就想把两个vue项目合并到一个域名下面.思考:vue的页面都是单页面应用,说白了就是一个index.h ...

  10. git 忽略文件不起作用

    本人需要提交项目文件,发现总有一些东西不需要提交,然后搜索有”.gitignore”文件可以忽略一些提交,但是发现添加上没有起作用. 要贴的是: /build/ target/ .idea/ *.im ...