【LinkedList】| 深度剥析Java SE 源码合集Ⅰ

news2024/12/25 12:51:41

目录

  • 一. 🦁 LinkedList介绍
  • 二. 🦁 结构以及对应方法分析
    • 2.1 结构组成
      • 2.1.1 节点类
      • 2.1.2 成员变量
    • 2.2 方法实现
      • 2.2.1 添加add(E e)方法
      • 2.2.2 头尾添加元素
        • Ⅰ addFirst(E e)
        • Ⅱ addLast(E e)
      • 2.2.3 查找get(int index)方法
      • 2.2.4 删除remove()方法
  • 三. 🦁 总结

一. 🦁 LinkedList介绍

LinkedList 底层用双向链表实现的存储。特点:查询效率低,增删效率高,线程不安全
双向链表也叫双链表,是链表的一种,它的每个数据节点中都有两个指针,分别指向前
一个节点和后一个节点。 所以,从双向链表中的任意一个节点开始,都可以很方便地找到所有节点。
在这里插入图片描述

今天来探索一下LinkedList的源码分析,以便更熟悉Java容器的结构,了解其是如何存储元素的。
探索环境:jdk 11 & idea 2020

二. 🦁 结构以及对应方法分析

2.1 结构组成

由源码可知,LinkedList 不仅继承了AbstractSequentialList,还实现了List,Deque等接口。所以LinkedList除了可以当做链表来操作外,它还可以当做栈、队列和双端队列来使用。
在这里插入图片描述

2.1.1 节点类

双向链表由节点类前后连接而成,所以源码中肯定存在该Node类。我们从源码中得知其节点类组成:

item:存储当前节点元素的信息
next:存储当前节点的下一个节点地址信息
prev:存储当前节点的前一个节点地址信息

  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;
        }
    }

2.1.2 成员变量

这里记录了LinkedList的节点数(size),头节点地址(first),尾节点(last)。

 transient int size = 0;

    /**
     * Pointer to first node.
     */
    transient Node<E> first;

    /**
     * Pointer to last node.
     */
    transient Node<E> last;

2.2 方法实现

2.2.1 添加add(E e)方法

在这里插入图片描述
从这里可知,这个add() 方法调用了linkLast(e)方法,返回一个布尔值。现在我们来看看这个linkLast(e)方法:

  /**
     * Links e as last element.
     */
    void linkLast(E e) {
        final Node<E> l = last;
        final Node<E> newNode = new Node<>(l, e, null);
        last = newNode;
        if (l == null)
            first = newNode;
        else
            l.next = newNode;
        size++;
        modCount++;
    }

从这个方法来看,add()方法的插入是尾插法,将last这个成员变量赋值给l(即l指向最当前节点),新new一个节点变量newNode<>(l,e,null),并且将其前驱节点指向l,如果 l == null,则第一个节点是newNode;否则直接在当前节点l指向新节点newNode。(modCount是指修改次数)。

在这里插入图片描述

2.2.2 头尾添加元素

Ⅰ addFirst(E e)

我们可以看到这里直接调用了linkFirst(E e)方法。
在这里插入图片描述

 /**
     * Links e as first element.
     */
    private void linkFirst(E e) {
        final Node<E> f = first;
        final Node<E> newNode = new Node<>(null, e, f);
        first = newNode;
        if (f == null)
            last = newNode;
        else
            f.prev = newNode;
        size++;
        modCount++;
    }

这里是运用了头插法,将节点插入链表头部,依旧f和前面的l一样,指向当前节点(第一个节点);然后new 一个新的Node<>(null,e,f),将新节点的后继节点指向当前节点。当前节点不为null的情况下,将当前节点的前驱节点指向新节点,这样一次插入完成。

在这里插入图片描述

Ⅱ addLast(E e)

尾插法同前面的add(E e)。

2.2.3 查找get(int index)方法

在这里插入图片描述
这里调用了两个方法,一个是 checkElementIndex(index),该方法是用来检查用户输入的下标是否存在,如果不合法的话则会抛出 **IndexOutOfBoundsException(outOfBoundsMsg(index))**异常,具体实现如图:
在这里插入图片描述
在这里插入图片描述
第二个方法是:
在这里插入图片描述
该方法很明显是用来返回获取对应下标元素的值,具体实现如下:

 /**
     * Returns the (non-null) Node at the specified element 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是小于一半LinkedList长度时,则从头节点开始遍历查找;相反如果index大于一半LinkedList长度时,则从尾节点开始查找,这也是双向链表的一个优点。

2.2.4 删除remove()方法

在这里插入图片描述
LinkedList的删除方法比较多,我们就来探索一个常用的remove(),由源码可知,这里调用了removeFirst()方法:

 /**
     * Removes and returns the first element from this list.
     *
     * @return the first element from this list
     * @throws NoSuchElementException if this list is empty
     */
    public E removeFirst() {
        final Node<E> f = first;
        if (f == null)
            throw new NoSuchElementException();
        return unlinkFirst(f);
    }

这里定义了f——>头节点,如果头节点为空,说明f为空,则说该LinkedList没有任何元素,则返回一个NoSuchElementException()错误。否则调用了unlinkFirst(f)方法。

 /**
     * Unlinks non-null first node f.
     */
    private E unlinkFirst(Node<E> f) {
        // assert f == first && f != null;
        final E element = f.item;
        final Node<E> next = f.next;
        f.item = null;
        f.next = null; // help GC
        first = next;
        if (next == null)
            last = null;
        else
            next.prev = null;
        size--;
        modCount++;
        return element;
    }

这里先定义了一个next节点指向头节点的下一个节点,然后赋值头节点的元素和next指针为null(这里主要是减少GC垃圾回收的工作量,提高效率),然后让头节点的指针指向next节点,这样next节点则成为新的一个头节点。就删除了头节点啦,size–,返回element。

三. 🦁 总结

今天介绍了LinkedList源码剥析。分析了常用方法的源码结构组成,LinkedList的源码组成比较简单,只要对双向链表这一数据结构熟悉的话,阅读起源码还是非常轻松的,希望您喜欢,一键三连哦!🐇 😄 🐇

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/389340.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【云原生】整合K8s + SpringCloudK8s + gRpc + RocketMQ + Istio + Envoy

背景本文把前面的代码整理一遍&#xff0c;不仅仅是demo层面&#xff0c;而是考虑到放进生产中使用&#xff0c;且尽可能用高版本&#xff0c;关于这块技术&#xff0c;网上的文章真是一言难尽&#xff0c;要么就是个概念&#xff0c;要么就是把官网的demo拿过来跑一遍&#xf…

图神经网络 pytorch GCN torch_geometric KarateClub 数据集

图神经网络 安装Pyg 首先安装torch_geometric需要安装pytorch然后查看一下自己电脑Pytorch的版本 import torch print(torch.__version__) #1.12.0cu113然后进入官网文档网站 链接: https://pytorch-geometric.readthedocs.io/en/latest/install/installation.html 安装自己…

【小破站下载工具】Python tkinter 实现网站下载工具,所有数据一键获取

目录前言开发环境本次项目案例步骤先展示下完成品的效果界面导入模块先创建个窗口功能按键主要功能代码编写功能一功能二功能三前言 最近很多同学想问我&#xff0c;怎么把几个代码的功能集合到一起&#xff1f; 很简单&#xff0c;写一个界面就行了&#xff0c;想要哪个代码…

黑马程序员7

算数运算符重载 运算符重载概念&#xff1a;对已有的运算符重新进行定义&#xff0c;赋予其另一种功能&#xff0c;以适应不同的数据类型 加号运算符 通过自己写函数&#xff0c;实现两个对象相加属性后返回新的对象 两种方式重载 成员函数方式重载 全局函数重载 上来 perso…

Go 内存分布

Go内存分布方式在C中&#xff0c;每个值在内存中只占据一个内存块&#xff08;一段连续内存&#xff09;&#xff1b;但是&#xff0c;一些Go类型的值可能占据多个内存块。以后&#xff0c;我们称一个Go值分布在不同内存块上的部分为此值的各个值部&#xff08;value part&…

网络安全平台测试赛 easyphp(phar脏数据处理)

昨天的比赛&#xff0c;14.00-17.00.时间有点紧张&#xff0c;比赛期间没拿下来这道 &#x1f62d;非常痛苦&#xff0c;很顺畅的思路 一步步想下来&#xff0c;卡在最后一步末尾脏数据处理了&#xff0c;最后时间到了 没打通&#xff0c;还需多练 这里本地复现一下&#xff1…

linux 进程及调度基础知识

引用Linux进程管理专题Linux进程管理与调度-之-目录导航Linux下0号进程的前世(init_task进程)今生(idle进程)----Linux进程的管理与调度&#xff08;五&#xff09;蜗窝科技-进程管理郭健&#xff1a; Linux进程调度技术的前世今生之“前世”郭健&#xff1a; Linux进程调度技术…

1.7 古典概型问题类型一——随机取数问题

(1)我的答案&#xff1a;一、信息首先7个数字全不相同二、分析七个数字全不相同意味着每次取出来的数都不一样&#xff0c;然后每次取出后选择少一种&#xff0c;为简单排列不含10和1这意味从8个数里面选且可以重复为重复排列3.10恰好出现两次隐含着两个问题&#xff0c;第一&a…

备考考研2数学

进度说明&#xff0c;开始形成自己的复习进度说明&#xff01; 14:21 2023年2月28日星期二 武忠祥数学 截止目前&#xff0c;看完了01.高数基础01 14:21 2023年2月28日星期二 开始看02. 现在15:04 2023年2月28日星期二&#xff0c; 因为这2天的百度网盘不能进行解析了&…

初识HTML技术

文章目录一、为什么学习前端?二、第一个HTML文件VSCode三. HTML元素四. HTML页面一、为什么学习前端? 我们作为一个后端程序员&#xff0c;为什么还要学习前端&#xff0c;因为我们的终极目的是实现web开发&#xff0c;搭建网站&#xff0c;网站 前端 后端 比如我们随便…

最近几篇较好论文实现代码(附源代码下载)

《Towards Layer-wise Image Vectorization》(CVPR 2022) GitHub: github.com/ma-xu/LIVEInstallationWe suggest users to use the conda for creating new python environment.Requirement: 5.0<GCC<6.0; nvcc >10.0.git clone gitgithub.com:ma-xu/LIVE.gitcd LIVE…

一步一步学会给Fritzing添加元器件-丰富你的器件库

文章目录1、获取元器件文件2、单个添加元器件3、批量加入&#xff08;1&#xff09;、通过别人发布的bin文件加载&#xff08;2&#xff09;、终极大招&#xff08;拖&#xff09;4、制作自己器件文章出处&#xff1a; https://blog.csdn.net/haigear/article/details/12931545…

【C++】类和对象——六大默认成员函数

&#x1f3d6;️作者&#xff1a;malloc不出对象 ⛺专栏&#xff1a;C的学习之路 &#x1f466;个人简介&#xff1a;一名双非本科院校大二在读的科班编程菜鸟&#xff0c;努力编程只为赶上各位大佬的步伐&#x1f648;&#x1f648; 目录前言一、类的6个默认成员函数二、构造…

错误异常捕获

1、React中错误异常捕获 在 React 中&#xff0c;可以通过 Error Boundaries&#xff08;错误边界&#xff09;来捕获错误异常。Error Boundaries 是一种 React 组件&#xff0c;它可以在其子组件树的渲染期间捕获 JavaScript 异常&#xff0c;并且可以渲染出备用 UI。React 提…

802.11 service服务类型

802.11 serviceservice定义service分类按照模块分为两类按照功能分为六类数据传输相关服务分布式服务DS&#xff08;Distribution Service&#xff09;整合服务IS&#xff08;Integration Service&#xff09;关联&#xff08;association&#xff09;重关联&#xff08;reasso…

RAD 11.3 delphi和C++改进后新增、废弃及优化的功能

RAD 11.3 delphi和C改进后新增和废弃的功能 目录 RAD 11.3 delphi和C改进后新增和废弃的功能 一、版本RAD 11.3 delphi和C改进后新增功能 1、官方视频位置&#xff1a; 2、官方文档的链接位置&#xff1a; 二、版本RAD 11.3 delphi和C改进后废弃的功能 2.1、编译器不再使…

Eureka注册中心和Nacos注册中心详解以及Nacos与Eureka有什么区别?

目录&#xff1a;前言Eureka注册中心Nacos注册中心Nacos与Eureka有什么区别&#xff1f;前言提供接口给其它微服务调用的微服务叫做服务提供者&#xff0c;而调用其它微服务提供的接口的微服务则是服务消费者。如果服务A调用了服务B&#xff0c;而服务B又调用了服务C&#xff0…

【iOS】设置背景渐变色

drawRect函数 主要负责iOS的绘图操作&#xff0c;程序会自动调用此方法进行绘图。我在这个函数中绘制渐变背景色。 方法定义&#xff1a; -(void)drawRect:(CGRect)rect; 重写此方法&#xff0c;执行重绘任务-(void)setNeedsDisplay; 标记为需要重绘&#xff0c;异步调用dra…

Mysql开发

Mysql开发 可以使用MySQL直接存储文件吗&#xff1f; 可以使用 BLOB (binary large object)&#xff0c;用来存储二进制大对象的字段类型。 TinyBlob 255 值的长度加上用于记录长度的1个字节(8位) Blob 65K值的长度加上用于记录长度的2个字节(16位) MediumBlob 16M值的长度加…

vue-v-for列表渲染中key的作用

1.虚拟DOM中key的作用: key是点拟DON对象的标识&#xff0c;当状态中的数据发生变化时&#xff0c;Vue会根据【新数据】生成【新的虚拟DOM】,随后Vue进行【新虚拟DOM】与【旧虚拟DOM】的差异比较&#xff0c;比较规则如下 2.对比规则: 旧虚拟DOM中找到了与新虚拟DOM相同的ke…