【数据结构基础_链表】
1、链表的定义
链表与数组的区分:
数组是一块连续的内存空间,有了这块内存空间的首地址,就能直接通过索引计算出任意位置的元素地址。
数组最大的优势是支持通过索引快速访问元素,而链表就不支持。链表不一样,一条链表并不需要一整块连续的内存空间存储元素。
链表的元素可以分散在内存空间的天涯海角,通过每个节点上的 next, prev 指针,将零散的内存块串联起来形成一个链式结构。1)这样可以提高内存的利用效率,链表的节点不需要挨在一起,给点内存 new 出来一个节点就能用,操作系统会觉得这娃好养活2)另外一个好处,它的节点要用的时候就能接上,不用的时候拆掉就行了,从来不需要考虑扩缩容和数据搬移的问题弊端:
因为元素并不是紧挨着的,所以如果你想要访问第 3 个链表元素,你就只能从头结点开始往顺着 next 指针往后找,直到找到第 3 个节点才行。
二、链表的类型
1、单链表
编程语言标准库一般都会提供泛型,即你可以指定 val 字段为任意类型,而力扣的单链表节点的 val 字段只有 int 类型。
class ListNode:def __init__(self, x):self.val = xself.next = None
2、双链表
编程语言标准库一般使用的都是双链表而非单链表。
单链表节点只有一个 next 指针,指向下一个节点;
而双链表节点有两个指针,prev 指向前一个节点,next 指向下一个节点。
class Node:def __init__(self, prev, element, next):self.val = elementself.next = next #指向下个元素的指针self.prev = prev #指向上个元素的指针
三、单链表的操作
首先,要创建一个单链表,来用于下面的操作
class ListNode:def __init__(self, x): # 修正了 __int__ 为 __init__self.val = xself.next = Nonedef createlinkedlist(arry: 'List[int]') -> ListNode:if arry is None or len(arry) == 0:return Nonehead = ListNode(arry[0]) # 创建头节点current = head # 使用一个指针来遍历链表for i in range(1, len(arry)):current.next = ListNode(arry[i]) # 创建新节点并链接current = current.next # 移动指针return head # 返回链表的头节点
1、对节点进行赋值,必须转化为ListNode类型head = ListNode(arry[0]) # 创建头节点current.next = ListNode(arry[i]) # 创建新节点并链接错误写法:current.next = arry[i]2、必须使用指针来遍历链表head = ListNode(arry[0]) # 创建头节点
current = head # 使用一个指针来遍历链表问题:为什么需要使用指针(如 current)?
1、保持链表头节点的引用
链表的头节点是链表的入口点,通常需要保持对它的引用,以便后续可以访问整个链表。如果不使用额外的指针,直接操作 head,可能会导致以下问题:
1)丢失链表头节点:在链表操作过程中,如果直接修改 head,可能会意外地丢失对链表的引用,导致无法再访问链表的其他部分。
2)无法返回链表头节点:在函数中创建链表时,最终需要返回链表的头节点。如果不使用额外的指针,直接操作 head,可能会导致返回的头节点指向错误的位置。2、方便链表的遍历和操作
使用指针(如 current)可以方便地遍历链表,并在遍历过程中对链表进行操作(如插入、删除节点)。指针的作用类似于一个“游标”,可以在不改变链表头节点的情况下,逐个访问链表的节点。3、如果不使用指针:
def createlinkedlist(arry: 'List[int]') -> ListNode:。。。head = ListNode(arry[0])for i in range(1, len(arry)):head.next = ListNode(arry[i])head = head.next # 直接修改 headreturn head # 返回的是最后一个节点,而不是头节点
运行结果:

1、查
# 创建一条单链表
head = createLinkedList([1, 2, 3, 4, 5])# P为指针,遍历单链表
p = head
while p is not None:print(p.val)p = p.next
2、增
在头部增加新节点
# 创建一条单链表
head = createLinkedList([1, 2, 3, 4, 5])# 在单链表头部插入一个新节点 0
newHead = ListNode(0)newHead.next = head
head = newHead
# 现在链表变成了 0 -> 1 -> 2 -> 3 -> 4 -> 5
在尾部增加新节点
# 创建一条单链表
head = createLinkedList([1, 2, 3, 4, 5])# 在单链表尾部插入一个新节点 6
p = head
# 先走到链表的最后一个节点
while p.next is not None:p = p.next
# 现在 p 就是链表的最后一个节点
# 在 p 后面插入新节点
p.next = ListNode(6)# 现在链表变成了 1 -> 2 -> 3 -> 4 -> 5 -> 6
在单链表中间插入新元素
# 创建一条单链表
head = createLinkedList([1, 2, 3, 4, 5])# 在第 3 个节点后面插入一个新节点 66
# 先要找到前驱节点,即第 3 个节点
p = head
for _ in range(2):p = p.next
# 此时 p 指向第 3 个节点
# 组装新节点的后驱指针
new_node = ListNode(66)
new_node.next = p.next# 插入新节点
p.next = new_node# 现在链表变成了 1 -> 2 -> 3 -> 66 -> 4 -> 5
3、删
删除一个节点,首先要找到要被删除节点的前驱节点,然后把这个前驱节点的 next 指针指向被删除节点的下一个节点。这样就能把被删除节点从链表中摘除了。
# 创建一条单链表
head = createLinkedList([1, 2, 3, 4, 5])# 删除第 4 个节点,要操作前驱节点
p = head
for i in range(2):p = p.next# 此时 p 指向第 3 个节点,即要删除节点的前驱节点
# 把第 4 个节点从链表中摘除
p.next = p.next.next# 现在链表变成了 1 -> 2 -> 3 -> 5
在单链表尾部删除元素
# 创建一条单链表
head = createLinkedList([1, 2, 3, 4, 5])# 删除尾节点
p = head
# 找到倒数第二个节点
while p.next.next is not None:p = p.next# 此时 p 指向倒数第二个节点
# 把尾节点从链表中摘除
p.next = None# 现在链表变成了 1 -> 2 -> 3 -> 4
在单链表头部删除元素
# 创建一条单链表
head = createLinkedList([1, 2, 3, 4, 5])# 删除头结点
head = head.next# 现在链表变成了 2 -> 3 -> 4 -> 5
四、双链表的操作
class DoublyListNode:def __init__(self, x):self.val = xself.next = Noneself.prev = Nonedef createDoublyLinkedList(arr: List[int]) -> Optional[DoublyListNode]:if not arr:return Nonehead = DoublyListNode(arr[0])cur = head# for 循环迭代创建双链表for val in arr[1:]:#基于切片进行迭代new_node = DoublyListNode(val)cur.next = new_nodenew_node.prev = curcur = cur.nextreturn head
判断空列表:方式一:更加显式if arr is None or len(arr):方式二:更加简洁if not arr:在 Python 中,if not arr 是一种简洁的写法,用于检查一个可迭代对象(如列表、字符串、字典等)是否为空。它基于 Python 的布尔上下文(Boolean Context):如果 arr 是 None 或者是一个空列表([]),if not arr 的条件为 True。如果 arr 是一个非空列表(如 [1, 2, 3]),if not arr 的条件为 False。因此,if not arr 可以同时检查 arr 是否为 None 或者是否为空列表。0为false,非0 为true
1、查
对于双链表的遍历和查找,我们可以从头节点或尾节点开始,根据需要向前或向后遍历:
# 创建一条双链表
head = createDoublyLinkedList([1, 2, 3, 4, 5])
tail = None# 1、从头节点向后遍历双链表
p = head
while p:print(p.val)tail = pp = p.next# 2、从尾节点向前遍历双链表
p = tail
while p:print(p.val)p = p.prev
2、增
在链表头节点插入一个值
# 创建一条双链表
head = create_doubly_linked_list([1, 2, 3, 4, 5])# 在双链表头部插入新节点 0
new_head = DoublyListNode(0)
new_head.next = head
head.prev = new_head
head = new_head
# 现在链表变成了 0 -> 1 -> 2 -> 3 -> 4 -> 5
在链表尾部插入一个值
# 创建一条双链表
head = createDoublyLinkedList([1, 2, 3, 4, 5])# p是一个指针,初始化是从头节点开始
p = head
# 先走到链表的最后一个节点
while p.next is not None:p = p.next# 在双链表尾部插入新节点 6
newNode = DoublyListNode(6)
p.next = newNode
newNode.prev = p
# 更新尾节点引用
p = newNode# 现在链表变成了 1 -> 2 -> 3 -> 4 -> 5 -> 6
在双链表中间插入元素
# 创建一条双链表
head = createDoublyLinkedList([1, 2, 3, 4, 5])# 在第 3 个节点后面插入新节点 66
# 找到第 3 个节点
p = head
for _ in range(2): #range(2)代表0,1p = p.next# 组装新节点
newNode = DoublyListNode(66)
newNode.next = p.next
newNode.prev = p# 插入新节点
p.next.prev = newNode
p.next = newNode# 现在链表变成了 1 -> 2 -> 3 -> 66 -> 4 -> 5
解释
_ 是一个特殊的变量名,通常被称为“占位符变量”for _ in range(2): 的作用
for _ in range(2): 的意思是:循环两次,每次循环中 _ 的值会从 range(2) 中依次取值(即 0 和 1),但 _ 的值在循环体中不会被使用。
3、删
在双链表中删除一个节点,需要调整前驱节点和后继节点的指针
# 创建一条双链表
head = createDoublyLinkedList([1, 2, 3, 4, 5])# 删除第 4 个节点
# 先找到第 3 个节点
p = head
for i in range(2):p = p.next# 现在 p 指向第 3 个节点,我们将它后面的那个节点摘除出去
toDelete = p.next# 把 toDelete 从链表中摘除
p.next = toDelete.next
toDelete.next.prev = p# 把 toDelete 的前后指针都置为 null 是个好习惯(可选)
toDelete.next = None
toDelete.prev = None# 现在链表变成了 1 -> 2 -> 3 -> 5
在双链表头部删除元素
# 创建一条双链表
head = createDoublyLinkedList([1, 2, 3, 4, 5])# 删除头结点
toDelete = head
head = head.next
head.prev = None# 清理已删除节点的指针
toDelete.next = None# 现在链表变成了 2 -> 3 -> 4 -> 5
在双链表尾部删除元素
# 创建一条双链表
head = createDoublyLinkedList([1, 2, 3, 4, 5])# 删除尾节点
p = head
# 找到尾结点
while p.next is not None:p = p.next# 现在 p 指向尾节点
# 把尾节点从链表中摘除
p.prev.next = None# 把被删结点的指针都断开是个好习惯(可选)
p.prev = None# 现在链表变成了 1 -> 2 -> 3 -> 4
防止内存泄漏:把删除的元素,赋值为None,就可以
相关文章:
【数据结构基础_链表】
1、链表的定义 链表与数组的区分: 数组是一块连续的内存空间,有了这块内存空间的首地址,就能直接通过索引计算出任意位置的元素地址。 数组最大的优势是支持通过索引快速访问元素,而链表就不支持。链表不一样,一条链…...
Java 实现 Redis中的GEO数据结构
Java 实现 Redis中的GEO数据结构 LBS (基于位置信息服务(Location-Based Service,LBS))应用访问的数据是和人 或物关联的一组经纬度信息,而且要能查询相邻的经纬度范围,GEO 就非常适合应用在 …...
PostgreSQL如何关闭自动commit
PostgreSQL如何关闭自动commit 在 PostgreSQL 中,默认情况下,每个 SQL 语句都会自动提交(即 AUTOCOMMIT 是开启的)。如果希望关闭自动提交,以便手动控制事务的提交和回滚,可以通过以下方法实现。 1 使用 …...
1、云原生写在前面
云原生技术是什么(包含哪些组件)?每个组件是负责什么?学习这些组件技术能解决什问题?哪些类企业需要用到? 这是标准系列的问题,通过 deepseek 的深度思考就能得到我们想要的易于理解的人话式的…...
Redis离线安装
Linux系统Centos安装部署Redis缓存插件 参考:Redis中文网: https://www.redis.net.cn/ 参考:RPM软件包下载地址: https://rpmfind.net/linux/RPM/index.html http://rpm.pbone.net/ https://mirrors.aliyun.com/centos/7/os…...
网络安全-攻击流程-应用层
应用层攻击针对OSI模型的第七层(应用层),主要利用协议漏洞、业务逻辑缺陷或用户交互弱点,直接威胁Web应用、API、数据库等服务。以下是常见应用层攻击类型及其流程,以及防御措施: 1. SQL注入(SQ…...
java八股文-spring
目录 1. spring基础 1.1 什么是Spring? 1.2 Spring有哪些优点? 1.3 Spring主要模块 1.4 Spring常用注解 1.5 Spring中Bean的作用域 1.6 Spring自动装配的方式 1.7 SpringBean的生命周期 1.8 多级缓存 1.9 循环依赖? 1 .8.1 原因 1.8…...
Jvascript网页设计案例:通过js实现一款密码强度检测,适用于等保测评整改
本文目录 前言功能预览样式特点总结:1. 整体视觉风格2. 密码输入框设计3. 强度指示条4. 结果文本与原因说明 功能特点总结:1. 密码强度检测2. 实时反馈机制3. 详细原因说明4. 视觉提示5. 交互体验优化 密码强度检测逻辑Html代码Javascript代码 前言 能满…...
【Scrapy】Scrapy教程2——工作原理
文章目录 数据流组件引擎Engine调度器Scheduler下载器Downloader爬虫Spiders项目管道Item Pipeline下载器中间件Downloader Middlewares爬虫中间件Spider Middlewares 在学习Scrapy前,我们需要先了解其架构和工作原理,这样才能很好的去使用Scrapy。 Scra…...
探索 DeepSeek:AI 领域的璀璨新星
在人工智能飞速发展的当下,DeepSeek 作为行业内的重要参与者,正以独特的技术和广泛的应用备受瞩目。 DeepSeek 是一家专注于实现 AGI(通用人工智能)的中国人工智能公司。它拥有自主研发的深度学习框架,能高效处理海量…...
宏基传奇swift edge偶尔开机BIOS重置
电脑是acer swift edge, SFA16-41,出厂是Win11系统, BIOS版本出厂1.04,更新到了目前最新1.10。 问题是 会偶尔开机ACER图标变小跑到屏幕左上方,下次开机BIOS就会被重置,开机等待很长时间。 因为是偶尔现象的…...
自动驾驶---如何打造一款属于自己的自动驾驶系统
在笔者的专栏《自动驾驶Planning决策规划》中,主要讲解了行车的相关知识,从Routing,到Behavior Planning,再到Motion Planning,以及最后的Control,笔者都做了相关介绍,其中主要包括算法在量产上…...
【C语言】第一期——数据类型变量常量
目录 1 字面量 2 整数类型 2.1 整数类型的取值范围 2.1.1 sizeof 运算符 2.2 GB、MB、KB、B之间的关系 2.3 定义整数类型的变量并打印 2.4 整数类型代码演示 3 浮点类型 3.1 浮点类型的取值范围 3.2 定义浮点类型变量并打印 3.3 保留2位小数点 4 char字符型 4.1…...
04运维实用篇(D4_日志)
目录 一、简介 二、代码中使用日志工具记录日志 1. 操作步骤 步骤1:添加日志记录操作 步骤2:设置日志输出级别 步骤3:设置日志组 2. 知识小结 三、优化日志对象创建代码 1. 实例 2. 总结 四、日志输出格式控制 1. 实例 2. 总结 …...
centos部署open-webui
提示:本文将简要介绍一下在linux下open-webui的安装过程,安装中未使用虚拟环境。 文章目录 一、open-webui是什么?二、安装流程1.openssl升级2.Python3.11安装3.sqlite安装升级4.pip 下载安装open-webui 总结 一、open-webui是什么? Open W…...
UE求职Demo开发日志#32 优化#1 交互逻辑实现接口、提取Bag和Warehouse的父类
1 定义并实现交互接口 接口定义: // Fill out your copyright notice in the Description page of Project Settings.#pragma once#include "CoreMinimal.h" #include "UObject/Interface.h" #include "MyInterActInterface.generated.h…...
Visonpro 检测是否有缺齿
一、效果展示 二、上面是原展开工具CogPolarUnwrapTool; 第二种方法: 用Blob 和 CogCopyRegionTool 三、 用预处理工具 加减常数,让图片变得更亮点 四、圆展开工具 五、模板匹配 六、代码分解 1.创建集合和文子显示工具 CogGraphicCollec…...
第1章大型互联网公司的基础架构——1.6 RPC服务
你可能在1.1节的引言中注意到业务服务层包括HTTP服务和RPC服务,两者的定位不一样。一般来说,一个业务场景的核心逻辑都是在RPC服务中实现的,强调的是服务于后台系统内部,所谓的“微服务”主要指的就是RPC服务;而HTTP服…...
今日AI和商界事件(2025-02-15)
根据2025年2月15日的科技动态,以下是今日AI领域的重要事件及相关进展总结: 1. DeepSeek日活突破3000万,开源生态加速AI普惠 里程碑意义:开源大模型DeepSeek宣布日活跃用户数突破3000万,其R1模型凭借开源策略和低成本优…...
算法题(69):搜索插入位置
审题: 需要我们在有序数组中找到等于target值的元素的下标若没有则返回target按顺序会插入的位置的索引 思路 : 我们可以使用二分查找的方法 方法一:二分查找 和普通的二分查找不同,本题若没有找到就需要返回它按顺序插入的位置的…...
Spark 之 入门讲解详细版(1)
1、简介 1.1 Spark简介 Spark是加州大学伯克利分校AMP实验室(Algorithms, Machines, and People Lab)开发通用内存并行计算框架。Spark在2013年6月进入Apache成为孵化项目,8个月后成为Apache顶级项目,速度之快足见过人之处&…...
边缘计算医疗风险自查APP开发方案
核心目标:在便携设备(智能手表/家用检测仪)部署轻量化疾病预测模型,实现低延迟、隐私安全的实时健康风险评估。 一、技术架构设计 #mermaid-svg-iuNaeeLK2YoFKfao {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg…...
【论文笔记】若干矿井粉尘检测算法概述
总的来说,传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度,通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...
vue3 定时器-定义全局方法 vue+ts
1.创建ts文件 路径:src/utils/timer.ts 完整代码: import { onUnmounted } from vuetype TimerCallback (...args: any[]) > voidexport function useGlobalTimer() {const timers: Map<number, NodeJS.Timeout> new Map()// 创建定时器con…...
Java面试专项一-准备篇
一、企业简历筛选规则 一般企业的简历筛选流程:首先由HR先筛选一部分简历后,在将简历给到对应的项目负责人后再进行下一步的操作。 HR如何筛选简历 例如:Boss直聘(招聘方平台) 直接按照条件进行筛选 例如:…...
【VLNs篇】07:NavRL—在动态环境中学习安全飞行
项目内容论文标题NavRL: 在动态环境中学习安全飞行 (NavRL: Learning Safe Flight in Dynamic Environments)核心问题解决无人机在包含静态和动态障碍物的复杂环境中进行安全、高效自主导航的挑战,克服传统方法和现有强化学习方法的局限性。核心算法基于近端策略优化…...
RabbitMQ入门4.1.0版本(基于java、SpringBoot操作)
RabbitMQ 一、RabbitMQ概述 RabbitMQ RabbitMQ最初由LShift和CohesiveFT于2007年开发,后来由Pivotal Software Inc.(现为VMware子公司)接管。RabbitMQ 是一个开源的消息代理和队列服务器,用 Erlang 语言编写。广泛应用于各种分布…...
GitHub 趋势日报 (2025年06月06日)
📊 由 TrendForge 系统生成 | 🌐 https://trendforge.devlive.org/ 🌐 本日报中的项目描述已自动翻译为中文 📈 今日获星趋势图 今日获星趋势图 590 cognee 551 onlook 399 project-based-learning 348 build-your-own-x 320 ne…...
代码规范和架构【立芯理论一】(2025.06.08)
1、代码规范的目标 代码简洁精炼、美观,可持续性好高效率高复用,可移植性好高内聚,低耦合没有冗余规范性,代码有规可循,可以看出自己当时的思考过程特殊排版,特殊语法,特殊指令,必须…...
淘宝扭蛋机小程序系统开发:打造互动性强的购物平台
淘宝扭蛋机小程序系统的开发,旨在打造一个互动性强的购物平台,让用户在购物的同时,能够享受到更多的乐趣和惊喜。 淘宝扭蛋机小程序系统拥有丰富的互动功能。用户可以通过虚拟摇杆操作扭蛋机,实现旋转、抽拉等动作,增…...
