LinkedList源码解析

1. 数据结构设计

(1) 节点结构

LinkedList 的核心是双向链表节点 Node

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;}
}
  • 双向链接:每个节点同时记录前驱和后继,支持双向遍历。
  • 首尾指针:通过 firstlast 字段快速访问链表两端。

(2) 与 ArrayList 的存储对比

特性LinkedListArrayList
底层结构双向链表动态数组
内存占用更高(每个元素额外存储指针)更低(连续内存)
CPU缓存友好性差(节点分散)好(局部性原理)

2. 核心操作实现

(1) 添加元素

尾部插入(add(E e)
public boolean add(E e) {linkLast(e); // 时间复杂度 O(1)return true;
}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++;
}
  • 优势:无需扩容,直接修改指针。
指定位置插入(add(int index, E element)
public void add(int index, E element) {checkPositionIndex(index);if (index == size) {linkLast(element); // 尾部插入优化} else {linkBefore(element, node(index)); // 需要遍历找到目标节点}
}Node<E> node(int 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;}
}
  • 时间复杂度
    • 头尾操作:O(1)
    • 中间操作:O(n)(需遍历)

(2) 删除元素

remove(int index)
public E remove(int index) {checkElementIndex(index);return unlink(node(index)); // 先找到节点,再解除链接
}E unlink(Node<E> x) {final E element = x.item;final Node<E> next = x.next;final Node<E> prev = x.prev;if (prev == null) {first = next; // 删除的是头节点} else {prev.next = next;x.prev = null; // 清除引用}if (next == null) {last = prev; // 删除的是尾节点} else {next.prev = prev;x.next = null;}x.item = null; // 帮助GCsize--;return element;
}
  • 关键点:正确处理头尾节点的边界条件。

(3) 访问元素

get(int index)
public E get(int index) {checkElementIndex(index);return node(index).item; // 需遍历链表
}
  • 性能缺陷:随机访问效率远低于 ArrayList(O(n) vs O(1))。

3. 迭代器实现

(1) 普通迭代器 (Iterator)

private class ListItr implements ListIterator<E> {private Node<E> lastReturned;private Node<E> next;private int nextIndex;private int expectedModCount = modCount;public boolean hasNext() { return nextIndex < size; }public E next() {checkForComodification();if (!hasNext()) throw new NoSuchElementException();lastReturned = next;next = next.next;nextIndex++;return lastReturned.item;}
}
  • 快速失败机制:通过 modCount 检测并发修改。

(2) 双向迭代器 (ListIterator)

支持向前遍历:

public E previous() {checkForComodification();if (!hasPrevious()) throw new NoSuchElementException();lastReturned = next = (next == null) ? last : next.prev;nextIndex--;return lastReturned.item;
}

4. 线程安全与性能优化

(1) 线程安全问题

  • 非线程安全:并发修改会导致数据不一致或迭代器失效。
  • 解决方案
    List<String> syncList = Collections.synchronizedList(new LinkedList<>());
    
    或使用并发容器 ConcurrentLinkedDeque

(2) 设计取舍

操作LinkedListArrayList
头部插入O(1)O(n)(需移动元素)
中间插入O(n)(查找)+ O(1)O(n)(移动元素)
随机访问O(n)O(1)
内存占用

5. 应用场景建议

  • 使用 LinkedList 的场景
    • 频繁在 头部/中部 插入删除(如实现队列/栈)。
    • 不需要频繁随机访问。
  • 使用 ArrayList 的场景
    • 需要快速随机访问。
    • 内存敏感型应用。

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

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

相关文章

语雀批量导出知识库

使用工具&#xff1a;yuque-dl 参考文档&#xff1a; GitHub - gxr404/yuque-dl: yuque 语雀知识库下载 Yuque-DL&#xff1a;一款强大的语雀资源下载工具_语雀文档怎么下载-CSDN博客

电子电气架构 --- 当前企业EEA现状(下)

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 做到欲望极简,了解自己的真实欲望,不受外在潮流的影响,不盲从,不跟风。把自己的精力全部用在自己。一是去掉多余,凡事找规律,基础是诚信;二是…

flink中的窗口的介绍

本文重点 无界流会源源不断的产生数据,有的时候我们需要把无界流进行切分成一段一段的有界数据,把一段内的所有数据看成一个整体进行聚合计算,这是实现无界流转成有界流的方式之一。 为什么需要窗口 数据是源源不断产生的,我们可能只关心某个周期内的统计结果。比如电费…

自建es 通过Flink同步mysql数据 Docker Compose

资源es:7.18 kibana:7.18 flink:1.17.2目录mkdir -p /usr/project/flink/{conf,job,logs} chmod -R 777 /usr/project/flink #资源情况 mysql8.0 Elasticsearch7.18 自建# 目录结构 /usr/project/flink/ /usr/project/flink/ ├── conf/ │ ├── flink-conf.yaml │ └…

AI浏览器和钉钉ONE是不是伪需求?

最近两则新闻格外引起了我的注意&#xff1a;一是Claude推出了官方浏览器插件&#xff0c;二是钉钉发布了钉钉ONE。前者说明AI浏览器未必有必要&#xff0c;后者则描绘了一幅“刷刷手机就能完成工作”的未来办公图景。这几天我经常在思考&#xff0c;AI浏览器是不是没有必要&am…

从结构化到多模态:RAG文档解析工具选型全指南

在RAG系统建设中&#xff0c;文档解析质量直接决定最终效果上限&#xff0c;选择合适的解析工具已成为避免"垃圾进&#xff0c;垃圾出"&#xff08;GIGO&#xff09;困境的关键决策。一、文档解析&#xff1a;RAG系统的基石与瓶颈 当前企业知识库中超过80%的信息存储…

设计模式:享元模式(Flyweight Pattern)

文章目录一、享元模式的介绍二、实例分析三、示例代码一、享元模式的介绍 享元模式&#xff08;Flyweight Pattern&#xff09; 是一种结构型设计模式。通过共享相同对象&#xff0c;减少内存消耗&#xff0c;提高性能。 它摒弃了在每个对象中保存所有数据的方式&#xff0c; 通…

【Go语言入门教程】 Go语言的起源与技术特点:从诞生到现代编程利器(一)

文章目录前言1. Go语言的起源与发展2. Go语言的核心设计团队2.1 Ken Thompson&#xff08;肯汤普森&#xff09;2.2 Rob Pike&#xff08;罗布派克&#xff09;2.3 Robert Griesemer&#xff08;罗伯特格瑞泽默&#xff09;设计动机&#xff1a;解决C的痛点3. Go语言的核心特性…

rocketmq启动与测试

1.更改runserver.sh的内存大小 vi runserver.sh 2.更改 runbroker.sh内存大小 vi runbroker.sh3.设置环境变量 vi ~/.bash_profile 新增 export NAMESRV_ADDRlocalhost:98764.启动 --在bin的上一级目录启动 nohup bin/mqnamesrv & nohup bin/mqbroker &5.查看日志 le…

11.《简单的路由重分布基础知识探秘》

11_路由重分布 文章目录11_路由重分布路由重分布概述路由重分布的核心作用基础实验实验流程实验拓扑配置示例(基本操作省略)实验结论路由重分布概述 路由重分布&#xff08;又称路由引入&#xff09;是指在不同路由协议之间交换路由信息的技术。在复杂网络中&#xff0c;可能同…

C++ 左值引用与右值引用介绍

C 左值引用与右值引用详解 在 C 的类型系统中&#xff0c;引用&#xff08;reference&#xff09; 是一种为已有对象起别名的机制。在早期&#xff08;C98/03&#xff09;中&#xff0c;C 只有 左值引用&#xff08;lvalue reference&#xff09;&#xff0c;主要用于函数参数…

基于物联网设计的园林灌溉系统(华为云IOT)_274

文章目录 一、前言 1.1 项目介绍 【1】项目开发背景 【2】设计实现的功能 【3】项目硬件模块组成 【4】设计意义 【5】国内外研究现状 【6】摘要 1.2 设计思路 1.3 系统功能总结 1.4 开发工具的选择 【1】设备端开发 【2】上位机开发 1.5 参考文献 1.6 系统框架图 1.7 系统原理…

uni-app iOS 应用版本迭代与上架实践 持续更新的高效流程

很多团队在使用 uni-app 开发 iOS 应用时&#xff0c;往往能顺利完成第一次上架&#xff0c;但一到 版本更新和迭代 环节&#xff0c;就会频繁遇到瓶颈&#xff1a;证书是否能复用&#xff1f;如何快速上传&#xff1f;怎样保持节奏不被打乱&#xff1f; 本文结合实战经验&…

解决由Tomcat部署前端改成nginx部署,导致大写.JPG结尾文件无法访问问题

前言&#xff1a;因信创替代要求&#xff0c;在麒麟服务器部署新的应用。原先的架构&#xff1a;前端tomcat部署&#xff0c;源码部署java应用&#xff08;ps&#xff1a;前后端&#xff0c;文件都在同一台服务器上&#xff09;&#xff0c;前端访问后端&#xff0c;再通过后端…

【设计模式】三大原则 单一职责原则、开放-封闭原则、依赖倒转原则

系列文章目录 文章目录系列文章目录一、单一职责原则方块游戏的设计二、开放-封闭原则原则介绍何时应对变化三、依赖倒转原则依赖倒转原则介绍里氏代换原则总结一、单一职责原则 单一职责原则&#xff0c;听字面意思&#xff0c;就是说功能要单一&#xff0c;他的准确解释是&a…

(3dnr)多帧视频图像去噪 (一)

一、多帧视频图像去噪 原理当摄像机每秒捕捉的图像达到60FPS&#xff0c;除了场景切换或者一些快速运动的场 景外&#xff0c;视频信号中相邻的两帧图像内容大部分是相同的。并且视频信号中的噪 声大部分都是均值为零的随机噪声&#xff0c;因此在时间上对视频信号做帧平均&…

从静态到智能:用函数式接口替代传统工具类

在 Java 早期开发中&#xff0c;我们习惯使用**静态实用程序类&#xff08;Utility Class&#xff09;**来集中放置一些通用方法&#xff0c;例如验证、字符串处理、数学计算等。这种模式虽然简单直接&#xff0c;但在现代 Java 开发&#xff08;尤其是 Java 8 引入 Lambda 和函…

免杀伪装 ----> R3进程伪装实战(高阶) ---->培养红队免杀思路

目录 R3进程伪装(免杀技术)高阶技术说明 深入剖析Windows进程规避免杀技术 学习R3进程伪装的必备技能 R3进程伪装的核心知识点与实现步骤 核心知识点 实现步骤 免杀实现步骤 PEB与EPROCESS的深入解析 1. PEB&#xff08;进程环境块&#xff09; 2. EPROCESS 3. PEB与…

深度学习——基于卷积神经网络实现食物图像分类(数据增强)

文章目录 引言 一、项目概述 二、环境准备 三、数据预处理 3.1 数据增强与标准化 3.2 数据集准备 四、自定义数据集类 五、构建CNN模型 六、训练与评估 6.1 训练函数 6.2 评估函数 6.3 训练流程 七、关键技术与优化 八、常见问题与解决 九、完整代码 十、总结 引言 本文将详细介…

【开题答辩全过程】以 基于微信小程序的教学辅助系统 为例,包含答辩的问题和答案

个人简介一名14年经验的资深毕设内行人&#xff0c;语言擅长Java、php、微信小程序、Python、Golang、安卓Android等开发项目包括大数据、深度学习、网站、小程序、安卓、算法。平常会做一些项目定制化开发、代码讲解、答辩教学、文档编写、也懂一些降重方面的技巧。感谢大家的…