【TypeScript】常见数据结构与算法(二):链表
文章目录
- 链表结构(LinkedList)
- 链表以及数组的缺点
- 数组
- 链表的优势
- 什么是链表?
- 封装链表相关方法
- 源码
- 链表常见面试题
- 237-删除链表中的节点
- 206 - 反转链表
- 数组和链表的复杂度对比
链表结构(LinkedList)
链表以及数组的缺点
- 链表和数组一样,可以用于存储一系列的元素,但是链表和数组的实现机制完全不同。
- 这一章中,我们就来学习一下另外一种非常常见的用于存储数据的线性结构:链表。
数组
- 要存储多个元素,数组(或称为链表)可能是最常用的数据结构。
- 我们之前说过,几乎每一种编程语言都有默认实现数组结构。
但是数组也有很多缺点
- 数组的创建通常需要申请一段连续的内存空间(一整块的内存),并且大小是固定的(大多数编程语言数组都是固定的),所以当当前数组不能满足容量需求时,需要扩容。(一般情况下是申请一个更大的数组,比如2倍。然后将原数组中的元素复制过去
- 而且在数组开头或中间位置插入数据的成本很高,需要进行大量元素的位移。
- 尽管JavaScript的Array底层可以帮我们做这些事,但背后的原理依然是这样。
链表的优势
要存储多个元素,另外一个选择就是链表
但不同于数组,链表中的元素在内存中不必是连续的空间。
- 链表的每个元素有一个存储元素本身的节点和一个指向下一个元素的引用(有些语言称为指针或者链接)组成。
相对于数组,链表的优势:
-
内存空间不是必须连续的。
√可以充分利用计算机的内存,实现灵活的内存动态管理。
-
链表不必再创建时就确定大小,并且大小可以无限的延伸下去。
-
链表在插入和删除数据时,时间复杂度可以达到O(1)
√相对数组效率高很多
相对数组,链表的缺点:
- 链表访问任何一个位置的元素时,都要从头开始访问。(无法跳过第一个元素访问任何一个元素)。
- 无法通过下标直接访问元素,需要从头一个个访问,直到找到对应元素。
什么是链表?
- 其实上面我们已经简单的提过了链表的结构,我们这里更加详细的分析一下。
- 链表类似于火车:有一个火车头,火车头会连接一个节点,节点上有乘客(类似于数据),并且这个节点会连接下一个节点,以此类推。


封装链表相关方法
我们先来认识一下,链表中应该有哪些常见的操作
append(element):向链表尾部添加一个新的项
insert(position,element):向链表的特定位置插入一个新的项。

get(position):获取对应位置的元素

indexOf(element):返回元素在链表中的索引。如果链表中没有该元素则返回-1。
update(position,element):修改某个位置的元素
removeAt(position):从链表的特定位置移除一项。

remove(element):从链表中移除一项。
isEmpty():如果链表中不包含任何元素,返回true,如果链表长度大于0则返回false。
size():返回链表包含的元素个数。与数组的length属性类似。
源码
// 1.创建Node节点类
class Node<T> {value: T;next: Node<T> | null = null;constructor(value: T) {this.value = value;}
}// 2.创建LinkedList的类
class LinkedList<T> {private head: Node<T> | null = null;private size: number = 0;get length() {return this.size;}// 封装私有方法// 根绝positon获取当前的节点(不是节点的value,而是获取节点)private getNode(position: number): Node<T> | null {let index = 0;let current = this.head;while (index++ < position && current) {current = current.next;}return current;}// 追加节点append(value: T) {// 1.根据value新建一个Node节点const newNode = new Node(value);// 2.if (!this.head) {this.head = newNode;} else {let current = this.head;while (current.next) {current = current.next;}// current肯定指向最后一个节点current.next = newNode;}// 3.size++this.size++;}// 遍历链表的方法traverse() {const values: T[] = [];let current = this.head;while (current) {values.push(current.value);current = current.next;}console.log(values.join(" -> "));}//链表插入元素的方法insert(value: T, position: number): boolean {// 1.越界判断if (position < 0 || position > this.size) return false;// 2.根据value创建新的节点const newNode = new Node(value);/* 3.判断* 是否插入头部* 否则就找到需要插入的位置,然后记录前一个节点和当前节点,在前一个节点的next等于newNode,newNode的next等于后一个节点*/if (position === 0) {newNode.next = this.head;this.head = newNode;} else {const previous = this.getNode(position - 1);newNode.next = previous!.next;previous!.next = newNode;}this.size++;return true;}//链表插入元素的方法removeAt(position: number): T | null {// 1.越界判断if (position < 0 || position >= this.size) return null;let current = this.head;if (position === 0) {this.head = current?.next ?? null;} else {const previous = this.getNode(position - 1);previous!.next = previous?.next?.next ?? null;}this.size--;return current?.value ?? null;}// 获取方法get(position: number): T | null {if (position < 0 || position >= this.size) return null;return this.getNode(position)?.value ?? null;}// 更新方法update(value: T, position: number): boolean {if (position < 0 || position >= this.size) return false;const currentNode = this.getNode(position);currentNode!.value = value;return true;}// 根据值获取该值位置索引indexOf(value: T): number {let current = this.head;let index = 0;while (current) {if (current.value === value) {return index;}index++;current = current.next;}return -1;}// 删除方法,根据value删除remove(value: T): T | null {const index = this.indexOf(value);return this.removeAt(index);}// 判断单链表是否为空的方法isEmpty() {return this.size === 0;}
}export {};
链表常见面试题
237-删除链表中的节点
- https://leetcode.cn/problems/delete-node-in-a-linked-list/

解题
// Definition for singly-linked list.
class ListNode {val: number;next: ListNode | null;constructor(val?: number, next?: ListNode | null) {this.val = val === undefined ? 0 : val;this.next = next === undefined ? null : next;}
}/**Do not return anything, modify it in-place instead.*/
function deleteNode(node: ListNode | null): void {node!.val = node!.next!.valnode!.next = node!.next!.next
}
206 - 反转链表
- https://leetcode.cn/problems/reverse-linked-list/

- 用栈结构解决
function reverseList(head: ListNode | null): ListNode | null {if(head === null) return nullif(head.next === null) return headconst stack:ListNode[] = []let current:ListNode | null = headwhile(current) {stack.push(current)current = current.next}const newHead:ListNode = stack.pop()!let newHeadCurrent = newHeadwhile(stack.length) {const node = stack.pop()!newHeadCurrent.next = nodenewHeadCurrent = newHeadCurrent.next}newHeadCurrent.next = nullreturn newHead
};
- 用非递归解题
function reverseList(head: ListNode | null): ListNode | null {if (head === null || head.next === null) return head;let newHead: ListNode | null = null;// 1.next = 2, 2.next = 3, 3.next = 4while (head) {// 让current指向下一个节点// 目的:保留下个节点的引用,可以拿到,并且不会销毁(current = 2)const current= head.next;// 改变head当前指向的节点,指向newHead// 这里反转链表对于第一节点来说,指向newHead就是null(1.next = null)head.next = newHead;// 让newhead指向head节点// 这里开始准备反转新的节点,目的是下一次遍历时,可以让下一个节点指向第一个节点(newHead = 1)newHead = head;// 让head指向下个节点也就是current(head = 2)head = current;}return newHead;
}

- 用递归方案解题
function reverseList(head: ListNode | null): ListNode | null {// 递归停止条件,当递归到最后一个节点时停止if (head === null || head.next === null) return head;// 一直递归循环直到符合head === null 时停止递归// 那么拿到的就是链表倒数第二个节点const newHead = reverseList(head.next ?? null)// 反转链表,让最后一个节点指向head开始正式反转head.next.next = head// 让倒数第二个节点的next指向nullhead.next = null// 最后递归完了就是反转后的链表了return newHead
}
数组和链表的复杂度对比
接下来,我们使用大O表示法来对比一下数组和链表的时间复杂度:

-
数组是一种连续的存储结构,通过下标可以直接访问数组中的任意元素。
-
时间复杂度:对于数组,随机访问时间复杂度为o(1),插入和删除操作时间复杂度为o(n)。
-
空间复杂度:数组需要连续的存储空间,空间复杂度为o(n)。
-
链表是一种链式存储结构,通过指针链接起来的节点组成,访问链表中元素需要从头结点开始遍历。
-
时间复杂度:对于链表,随机访问时间复杂度为o(n),插入和删除操作时间复杂度为o(1)。
-
空间复杂度:链表需要为每个节点分配存储空间,空间复杂度为O(n)。
-
在实际开发中,选择使用数组还是链表需要根据具体应用场景来决定。
-
如果数据量不大,且需要频繁随机访问元素,使用数组可能会更好。
-
如果数据量大,或者需要频繁插入和删除元素,使用链表可能会更好。
相关文章:
【TypeScript】常见数据结构与算法(二):链表
文章目录 链表结构(LinkedList)链表以及数组的缺点数组链表的优势 什么是链表?封装链表相关方法源码链表常见面试题237-删除链表中的节点206 - 反转链表 数组和链表的复杂度对比 链表结构(LinkedList) 链表以及数组的缺点 链表…...
原型模式 (Prototype Pattern)
定义: 原型模式(Prototype Pattern)是一种创建型设计模式,它用于创建重复的对象,同时保持性能。这种模式的核心思想是通过复制一个已存在的实例来创建新的实例,而不是新建实例并对其进行初始化。原型模式适…...
项目总结报告(案例模板)
软件项目总结报告模板套用: 项目概要项目工作分析经验与教训改进建议可纳入的项目过程资产 --------进主页获取更多资料-------...
C++ Qt QByteArray用法介绍
作者:令狐掌门 技术交流QQ群:675120140 csdn博客:https://mingshiqiang.blog.csdn.net/ 文章目录 一、QByteArray的基本用法1、初始化和赋值2、访问和修改元素3、 常用方法4、数据转换二、QByteArray与文件操作三、QByteArray与网络编程四、QByteArray数据编码1、Base64 编解…...
蓝桥杯物联网竞赛_STM32L071_3_Oled显示
地位: 对于任何一门编程语言的学习,print函数毫无疑问是一种最好的调试手段,调试者不仅能通过它获取程序变量的运行状态而且通过对其合理使用获取程序的运行流程,更能通过关键变量的输出帮你验证推理的正确与否,朴素的…...
python-opencv轮廓检测(外轮廓检测和全部轮廓检测,计算轮廓面积和周长)
python-opencv轮廓检测(外轮廓检测和全部轮廓检测,计算轮廓面积和周长) 通过cv2.findContours,我们可以进行轮廓检测,当然也有很多检测模式,我们可以通过选择检测模式,进行外轮廓检测ÿ…...
LeetCode [简单] 1. 两数之和
给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并返回它们的数组下标。 你可以假设每种输入只会对应一个答案。但是,数组中同一个元素在答案里不能重复出现。 你可以按任意顺序返回…...
C++设计模式之工厂模式(下)——抽象工厂模式
抽象工厂模式 介绍示例示例使用运行结果抽象工厂模式的优缺点优点缺点 总结 介绍 抽象工厂模式是一种创建型设计模式,它提供了一种封装一组相关或相互依赖对象的方式,而无需指定它们具体的类。它允许客户端使用抽象接口来创建一系列相关的对象ÿ…...
2023亚太杯数学建模A题思路分析 - 采果机器人的图像识别技术
1 赛题 问题A 采果机器人的图像识别技术 中国是世界上最大的苹果生产国,年产量约为3500万吨。与此同时,中国也是世 界上最大的苹果出口国,全球每两个苹果中就有一个,全球超过六分之一的苹果出口 自中国。中国提出了一带一路倡议…...
关于Flink的旁路缓存与异步操作
1. 旁路缓存 1. 什么是旁路缓存? 将数据库中的数据,比较经常访问的数据,保存起来,以减少和硬盘数据库的交互 比如: 我们使用mysql时 经常查询一个表 , 而这个表又一般不会变化,就可以放在内存中,查找时直接对内存进行查找,而不需要再和mysql交互 2. 旁路缓存例子使用 dim层…...
MyBatis-Plus的分页插件和乐观锁插件
MyBatis-Plus: 探索分页查询和乐观锁插件 在现代的Web应用开发中,高效的数据处理是不可或缺的一部分。MyBatis-Plus,作为MyBatis的增强版,提供了多种插件来简化和优化数据库操作。在这篇博客中,我们将重点介绍两个非常实用的插件…...
批量将本地N个英文Html文档进行中文翻译-操作篇
Unity3D特效百例案例项目实战源码Android-Unity实战问题汇总游戏脚本-辅助自动化Android控件全解手册再战Android系列Scratch编程案例软考全系列Unity3D学习专栏蓝桥系列ChatGPT和AIGC 👉关于作者 专注于Android/Unity和各种游戏开发技巧,以及各种资源分…...
解决cad找不到vcruntime140.dll的方法,实测有效的5个的方法
最近,我在使用CAD软件时遇到了一个困扰我已久的问题:由于找不到vcruntime140.dll文件而导致CAD无法正常运行。经过一番努力和尝试,我终于找到了解决这个问题的方法。那么,如何解决vcruntime140.dll丢失的问题呢?本文将…...
2023亚太杯数学建模C题:我国新能源电动汽车的发展趋势,思路模型代码
问题C 我国新能源电动汽车的发展趋势 赛题思路:获取思路见文末名片,第一时间更新 新能源汽车是指以先进技术原理、新技术、新结构的非常规汽车燃料为动力来源( 非常规汽车燃料指汽油、柴油以外的燃料),将先进技术进行汽车动力控制…...
英语学习-爆破音
英文爆破音有:[p],[b],[t],[d],[k],[g]。 同时爆破音的发音会根据前后音的不同,发音不同,具体如下: ⒈ [p],[b],[t],[d],[k],[g] 中的任何两个音素相邻时,前面的发不完全爆破音,后面的就要完全地爆破。如…...
【Vue】图片切换
上一篇: vue的指令 https://blog.csdn.net/m0_67930426/article/details/134599378?spm1001.2014.3001.5502 本篇所需要的指令有: v-on v-bind v-show <!DOCTYPE html> <html lang"en"> <head><meta charset"…...
C++模拟如何实现vector的方法
任意位置插入,insert的返回值为新插入的第一个元素位置的迭代器;因为插入可能会进行扩容,导致start的值改变,所以先定义一个变量保存pos与start的相对位置;判断是否需要扩容;从插入位置开始,将所…...
芯知识 | 混音播报语音芯片的优势:革新音频应用的新力量
随着科技的进步,语音芯片在各个领域的应用越来越广泛。而在众多语音芯片中,混音播报语音芯片以其独特的优势,正逐渐成为音频应用领域的翘楚。本文将重点探讨混音播报语音芯片的优势及其在现代科技应用中的价值。 一、混音播报语音芯片概述 …...
Arduino驱动PT100数字K型高温传感器(温湿度传感器)
目录 1、传感器特性 2、控制器和传感器连线图 3、硬件原理图 4、驱动程序 PT100适用于大部分400℃以下高温的测量,但是通常家用天然气灶焰芯温度可达800℃以上,烧制陶瓷的窖子或者大功率电炉温度更可超过1000℃,在这些超高温度的场景下就需要用到K型热电偶。...
【C/PTA —— 11.函数2(课外实践)】
C/PTA —— 11.函数2(课外实践) 一.函数题6-1 计算A[n]1/(1 A[n-1])6-2 递归实现顺序输出整数6-3 自然数的位数(递归版)6-4 分治法求解金块问题6-5 汉诺塔6-6 重复显示字符(递归版)6-7 显示平行四边形(右)(递归版) 二.编程题7-2 N阶楼梯上楼问题 一.函数…...
开拓药业销售业绩超预期 核心脱发新药KX-826进入上市前关键期
近日,开拓药业(09939.HK)密集发布2026年以来经营及销售成果公告,公司在美妆电商、海外业务、创新原料等板块均实现爆发式增长,商业化能力得到全面验证。随着核心脱发新药KX-826进入上市阶段,这家创新药企正…...
IMS放音信令机制:从183到UPDATE的早期媒体流控制
1. IMS放音信令机制的核心价值 想象一下你拨打电话时听到的"您拨打的用户正忙"提示音,这种看似简单的语音背后隐藏着一套精密的信令控制系统。在IMS网络中,早期媒体流(P-Early-Media)的传输质量直接影响用户体验&#x…...
AI写论文的秘密武器!4款AI论文生成工具,让论文写作更轻松!
在2025年,学术写作将迎来一场智能化的浪潮,越来越多的人开始尝试使用AI写论文工具。当面对硕士、博士论文这样的长篇力作时,很多工具却无法满足要求,有的缺乏深厚的理论基础,有的逻辑结构松散。普通的AI论文写作工具完…...
上海交通大学LaTeX论文模板:告别格式焦虑的学术写作终极指南
上海交通大学LaTeX论文模板:告别格式焦虑的学术写作终极指南 【免费下载链接】SJTUThesis 上海交通大学 LaTeX 论文模板 | Shanghai Jiao Tong University LaTeX Thesis Template 项目地址: https://gitcode.com/gh_mirrors/sj/SJTUThesis 你是否曾在深夜为论…...
科技赋能娱乐:超元力XR无轨黑暗乘骑的技术创新与体验革新
在科技与娱乐深度融合的当下,游乐产品的核心竞争力已从单纯的刺激感,转向沉浸式、互动性与创新性的综合体验。超元力XR无轨黑暗乘骑凭借全球首创的技术架构,将XR、AGV、动感控制等前沿技术与传统黑暗乘骑相结合,实现了技术与体验的…...
揭秘低查重AI教材编写秘籍,AI写教材工具助你高效完成专业教材!
在教材编写过程中,如何平衡原创性与合规性是一个新的挑战。许多创作者往往在借鉴优秀教材的内容时,难免担心查重率超出标准;而在尝试独立撰写知识点时,又会顾虑逻辑是否严谨、信息是否准确。更重要的是,当引用他人的研…...
VMware macOS虚拟机终极解锁指南:Unlocker完整使用教程
VMware macOS虚拟机终极解锁指南:Unlocker完整使用教程 【免费下载链接】unlocker VMware Workstation macOS 项目地址: https://gitcode.com/gh_mirrors/unloc/unlocker 在虚拟化技术日益普及的今天,你是否曾因VMware不支持macOS而苦恼…...
BN层真的是‘炼丹’万能药吗?聊聊我在小Batch Size和RNN上踩过的坑
BN层真的是‘炼丹’万能药吗?聊聊我在小Batch Size和RNN上踩过的坑 Batch Normalization(BN)自2015年提出以来,迅速成为深度学习模型中的标配组件。它被广泛认为能够加速训练、稳定梯度、降低对初始化的敏感度,甚至具备…...
PreScan泊车模型里的超声波传感器:参数怎么调?避坑指南来了
PreScan泊车模型中的超声波传感器参数调优实战指南 泊车辅助系统作为自动驾驶技术中最先落地的功能之一,其仿真验证环节直接关系到实际应用的安全性和可靠性。在PreScan仿真环境中,超声波传感器的参数配置往往成为影响整个泊车模型表现的关键变量。许多工…...
5步搞定MinGW-w64:在Windows上打造专业C/C++开发环境的终极指南
5步搞定MinGW-w64:在Windows上打造专业C/C开发环境的终极指南 【免费下载链接】mingw-w64 (Unofficial) Mirror of mingw-w64-code 项目地址: https://gitcode.com/gh_mirrors/mi/mingw-w64 你是否想在Windows系统上搭建一个功能完整、性能出色的C/C开发环境…...
