1、简介

LinkedList 底层使用的是 双向链表的数据结构

2、类图(JDK 1.8)

下图是LinkedList实现的接口和继承的类关系图:

public class LinkedList<E>
extends AbstractSequentialList<E>
implements List<E>, Deque<E>, Cloneable, java.io.Serializable

1、实现了四个接口

1.1 java.util.List 接口,提供数组的添加、删除、修改、迭代遍历等操作。

1.2 java.io.Serializable 接口,表示 LinkedList 支持序列化的功能。

1.3 java.lang.Cloneable 接口,表示 LinkedList 支持克隆。

1.4 java.util.Deque接口,是一种双向线性集合,可以再两端对集合进行插入或者删除操作,所以既可以当队列(FIFO),也可以当栈(LIFO),Deque继承java.util.Queue。

2、继承了一个类

2.1 java.util.AbstractSequentialList抽象类,它继承了AbstractList抽象类,并且有

public abstract ListIterator<E> listIterator(int index);抽象方法,返回一个由LinkedList实现的ListIterator迭代器,用来支持AbstractList抽象类的get(int index),set(int index, E element),add(int index, E element)remove(int index) 等一些支持随机访问的方法。

3、常用方法

添加元素 add(E e)

    //链表添加元素
public boolean add(E e) {
linkLast(e);//往链表尾部添加元素
return true;
}
void linkLast(E e) {
final Node<E> l = last;//临时变量l=last,后面会直接修改last节点 指向 新加入的节点
final Node<E> newNode = new Node<>(l, e, null);//新元素,创建成一个节点
last = newNode;//最后节点,指向新加入的节点
if (l == null)//判断last是否为空,为空代表是第一次新增节点
first = newNode;//首节点指向新加入的节点
else//last不为空表示当前链表有值,让l节点的参数 next指向新加入的节点
l.next = newNode;
size++;//链表大小+1
modCount++;//链表修改次数+1
} private static class Node<E> {
E item;//节点-值
Node<E> next;//下一个节点
Node<E> prev;//上一个节点 Node(Node<E> prev, E element, Node<E> next) {
this.item = element;
this.next = next;
this.prev = prev;
}
}

链表指定位置添加元素 add(int index, E element)

    public void add(int index, E element) {
checkPositionIndex(index);//判断index是否>=0,并且index<=size,否则抛出IndexOutOfBoundsException异常 if (index == size)//判断index等于size直接添加到链表尾部
linkLast(element);//同add(E e)方法
else
linkBefore(element, node(index));//node()方法先根据index标在链表取得对应到Node节点,并把新元素,添加到Node节点之前
} //根据下标获得节点
Node<E> node(int index) {
// assert isElementIndex(index); //size >> 1 等同 size除以2
//判断index 下标是在链表的前半部分,是则从链表头部开始遍历,否则从链表尾部向前遍历
//加了一个判断,最多只需要遍历一半链表即可,复杂度从O(n)降到了O(1)
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;
}
}

根据下标删除节点 remove(int index)

   //根据下标删除节点
public E remove(int index) {
checkElementIndex(index);//判断下标是否越界
return unlink(node(index));
} //链表-删除节点
E unlink(Node<E> x) {
// assert x != null;
final E element = x.item;//删除节点的元素
final Node<E> next = x.next;//删除节点的下一个节点
final Node<E> prev = x.prev;//删除节点的上一个节点 if (prev == null) {//prev等于null,说明是第一个节点
first = next;
} else {
prev.next = next;//删除节点的上一个节点,next参数指向 删除节点的下一个节点
x.prev = null;//删除节点的上一个节点,设置为null取消关联
} if (next == null) {//next等于null,说明是尾部节点
last = prev;
} else {
next.prev = prev;//删除节点的下一个节点,参数prev指向 删除节点的上一个节点
x.next = null;//删除节点的下一个节点,设置为null取消关联
} x.item = null;//节点元素设置为null,方便gc回收
size--;//链表大小-1
modCount++;//链表修改次数+1
return element;//返回删除的元素值
}

根据元素删除节点 remove(Object o)

    //删除链表中第一个匹配o的元素
public boolean remove(Object o) {
if (o == null) {//先判断o是否为空
for (Node<E> x = first; x != null; x = x.next) {
if (x.item == null) {//判断元素是否为null
unlink(x);//链表删除元素
return true;
}
}
} else {
for (Node<E> x = first; x != null; x = x.next) {
if (o.equals(x.item)) {//判断元素是否相等
unlink(x);//链表删除元素
return true;
}
}
}
return false;
}

根据下标获得元素 get(int index)

    //根据下标获得节点
public E get(int index) {
checkElementIndex(index);//判断index是大于0,并且小于等于size,否则抛出异常
return node(index).item;//node()方法获取下标对应的Node,并返回Node的item
}

时间复杂度

添加元素 add(E e)

最好时间复杂度是 O(1) ,最坏时间复杂度是 O(n) ,平均时间复杂度是 O(n) 。

最好时间复杂度发生在头部、或尾部添加的情况。

链表指定位置添加元素 add(int index, E element)

最好时间复杂度是 O(1) ,最坏时间复杂度是 O(n) ,平均时间复杂度是 O(n) 。

最好时间复杂度发生在头部、或尾部添加的情况。

根据下标删除节点 remove(int index)

最好时间复杂度是 O(1) ,最坏时间复杂度是 O(n) ,平均时间复杂度是 O(n) 。

最好时间复杂度发生在头部、或尾部移除的情况。

根据元素删除节点 remove(Object o)

最好时间复杂度是 O(1) ,最坏时间复杂度是 O(n) ,平均时间复杂度是 O(n) 。

最好时间复杂度发生在头部移除的情况。

根据下标获得元素 get(int index)

最好时间复杂度是 O(1),平均时间复杂度是 O(n) ,查找指定元素的平均时间复杂度是 O(n) 。

最好时间复杂度发生在下标为0或者等于size的位置,因为LinkedList会根据index是否超过了 size除以2 来作为依据看是从链表头部往后面遍历,还是链表尾部向前遍历,所以基本上只需要遍历一半的链表就能找到对应下标的元素了。

小结

1、LinkedList做插入、删除的时候,慢在寻址,快在只需要改变前后Node的引用地址即可

2、适用于频繁的元素修改的场景,对于查询场景比较多的适合请适用ArrayList这种数据结构

java集合-链表LinkedList的更多相关文章

  1. 【Java集合】LinkedList详解前篇

    [Java集合]LinkedList详解前篇 一.背景 最近在看一本<Redis深度历险>的书籍,书中第二节讲了Redis的5种数据结构,其中看到redis的list结构时,作者提到red ...

  2. Java集合:LinkedList源码解析

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

  3. 死磕 java集合之LinkedList源码分析

    问题 (1)LinkedList只是一个List吗? (2)LinkedList还有其它什么特性吗? (3)LinkedList为啥经常拿出来跟ArrayList比较? (4)我为什么把LinkedL ...

  4. java集合之linkedList链表基础

    LinkedList链表: List接口的链接列表实现.允许存储所有元素(包含null).使用频繁增删元素. linkedList方法: void addFirst(E e) 指定元素插入列表的开头 ...

  5. Java 集合之LinkedList源码分析

    1.介绍 链表是数据结构中一种很重要的数据结构,一个链表含有一个或者多个节点,每个节点处理保存自己的信息之外还需要保存上一个节点以及下一个节点的指针信息.通过链表的表头就可以访问整个链表的信息.Jav ...

  6. Java集合干货——LinkedList源码分析

    前言 在上篇文章中我们对ArrayList对了详细的分析,今天我们来说一说LinkedList.他们之间有什么区别呢?最大的区别就是底层数据结构的实现不一样,ArrayList是数组实现的(具体看上一 ...

  7. Java集合(五)--LinkedList源码解读

    首先看一下LinkedList基本源码,基于jdk1.8 public class LinkedList<E> extends AbstractSequentialList<E> ...

  8. Java集合:LinkedList (JDK1.8 源码解读)

    LinkedList介绍 还是和ArrayList同样的套路,顾名思义,linked,那必然是基于链表实现的,链表是一种线性的储存结构,将储存的数据存放在一个存储单元里面,并且这个存储单元里面还维护了 ...

  9. Java集合之LinkedList

    一.LinkedList概述 1.初识LinkedList 上一篇中讲解了ArrayList,本篇文章讲解一下LinkedList的实现. LinkedList是基于链表实现的,所以先讲解一下什么是链 ...

随机推荐

  1. 2、MyBatis教程之第一个MyBatis程序

    3.MyBatis第一个程序 1.搭建实验数据库 CREATE DATABASE `mybatis`; USE `mybatis`; DROP TABLE IF EXISTS `user`; CREA ...

  2. Android Studio 之生成正式签名的 APK 文件

    生成 APK 文件 •步骤 点击  Build -> Generate Signed...... : 来到如下界面: 选择 APK 选项,点击 Next 来到如下界面: 如果你电脑上没有一个正式 ...

  3. [树形DP]战略游戏

    战 略 游 戏 战略游戏 战略游戏 题目描述 Bob喜欢玩电脑游戏,特别是战略游戏.但是他经常无法找到快速玩过游戏的办法.现在他有个问题.他要建立一个古城堡,城堡中的路形成一棵树.他要在这棵树的结点上 ...

  4. cordova app打包apk签名

    首先执行:ionic cordova build android --prod --release,执行完会在以下目录生成apk文件( --prod 用以压缩) 然后使用keytool生成keysto ...

  5. 痞子衡嵌入式:从头开始认识i.MXRT启动头FDCB里的lookupTable

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT启动头FDCB里的lookupTable. 一个MCU内部通常有很多外设模块,这些外设模块是各MCU厂商做差异化产品的本质, ...

  6. MRCTF My secret

    My secret 知识点:wireshark基本操作,shadowsocks3.0源码利用,拼图(os脚本编写能力), 根据这里的信息可以知道,tcp所传输的源数据是在target address后 ...

  7. kernel base

    基础知识 学习网址:ctfwiki 安全客 Kernel:又称核心 维基百科:在计算机科学中是一个用来管理软件发出的数据I/O(输入与输出)要求的电脑程序,将这些要求转译为数据处理的指令并交由中央处理 ...

  8. redhat7.6 安装 Python 3

    redhat7.6 默认安装了python2.7. 查看已安装python情况 cd / # 定位到根目录 whereis python # 查看python所在目录 /usr/bin cd /usr ...

  9. ArrayList、CopyOnWriteArrayList源码解析(JDK1.8)

    本篇文章主要是学习后的知识记录,存在不足,或许不够深入,还请谅解. 目录 ArrayList源码解析 ArrayList中的变量 ArrayList构造函数 ArrayList中的add方法 Arra ...

  10. Java代理简述

    1.什么是代理? 对类或对象(目标对象)进行增强功能,最终形成一个新的代理对象,(Spring Framework中)当应用调用该对象(目标对象)的方法时,实际调用的是代理对象增强后的方法,比如对功能 ...