【2】数据结构的单链表章
目录标题
- 单链表的定义
- 单链表的初始化
- 单链表的建立
- 头插法创建
- 尾插法创建
- 查找操作
- 按序号查找
- 按内容查找
- 插入操作
- 删除操作
- 合并操作
- 单链表总代码与调试
单链表的定义
- 结点(Node)的定义:数据域(data)和指针域(next)。其中,
- data存储结点的值
- next存储直接后继的地址
- 代码声明结点
class Node:"""定义节点类型"""def __init__(self, data):# 存储数据元素的数据域self.data = data# 存储指向后继结点位置的指针域self.next = None
- 结点示意图

- 定义:链表中的每一个结点只有一个指针域,将该类链表称为单链表。以单链表(A,B,C,D,E)为例。
- 单链表逻辑结构示意图

- 单链表物理结构示意图

单链表的初始化
- 初始化单链表
def __init__(self):"""单链表的初始化"""# 声明头结点self.head = Node(None)
- 判断单链表是否为空表
def isEmpty(self):"""判断单链表是否为空表:return: True or False"""# 如果头结点的next域为空,则返回True,否则返回falsereturn self.head.next == None
- 求单链表的长度
def getLength(self):"""获取单链表的长度:return: 当前单链表中数据元素的个数"""# 设置length,用来计算长度,初始值为0length = 0# 声明cur指针,用于遍历单链表,初始值为第一个结点cur = self.head.next# 循环,直到cur==Nonewhile cur != None:# 单链表长度加1length += 1# cur指针指向当前结点的后继cur = cur.nextreturn length
- 打印单链表
def display(self):"""展示单链表:return:"""#判断是否为空表if self.isEmpty():print("当前单链表为空表!")return# 遍历单链表,并展示数据元素print("单链表中的数据元素:", end="")cur = self.head.nextwhile cur!= None:print(cur.data, end=",")cur = cur.nextprint()
单链表的建立
头插法创建
-
将需加入的结点,不断加入到头结点的后面。
-
核心思想:
- 将新结点的next指向头结点的后继结点
- 将头结点的next指向新结点
-
头插法示意图

-
代码定义头插法函数
def prepend(self, data):"""头插法:param data: 待插入的元素:return:"""# 创建新结点,在新结点中存储元素newNode = Node(data)# 修改指针指向,实现插入# 将新结点的next指向头结点的后继结点newNode.next = self.head.next# 将头结点的next指向新节点self.head.next = newNode
尾插法创建
-
将新结点不断地加入为当前链表的尾部。
-
核心思想:
- 寻找尾结点
- 将尾指针的next指向新结点
-
尾插法示意图

-
代码定义尾插法函数
def append(self, data):"""尾插法:param data: 待插入的元素:return:"""# 创建尾指针,初始化指向头结点rear = self.head# 循环,寻找尾结点while rear.next != None:rear = rear.next# 创建新结点,在新节点中存储元素newNode = Node(data)# 修改指针指向,实现插入# 将尾指针的next指向新结点rear.next = newNode
查找操作
按序号查找
- 从头结点开始扫描,直到第index个结点数,返回结点。
- 代码定义方法
def getData(self, index):"""按序号查找:param index: 待查找的序号:return: 待查找位置对应的元素值"""# 判断index是否合法if index <= 0 or index > self.getLength():raise IndexError('index 非法')# 声明变量作为计数器,累计当前的结点数。初始为0i = 0# 声明指针cur,用于遍历cur = self.head# 循环扫描while cur != None and i != index:cur = cur.nexti += 1return cur
按内容查找
-
在表中查找与内容值key相同的结点,找到了返回下标,否则返回空序号-1。
-
代码定义方法
def locate(self, key):"""按内容查找:param key: 待查找的内容:return: 查找内容的位置序号"""# 声明index为索引位置,初始-1index = -1cur = self.head.nexti = 0while cur != None:i += 1if cur.data == key:index = ibreakcur = cur.nextreturn index
插入操作
-
核心思想:
- 遍历找到需插入的前一个结点,并用pre指针指向它
- 将新结点的next指向pre指针所指向的结点的后续结点
- 将pre指针所指向的结点的next指向新结点
-
插入示意图

-
代码定义方法
def insert(self, index, data):"""在单链表中任意位置插入元素:param index: 待插入的位置:param data: 待插入的元素:return:"""# 判断index是否合法if index <= 0 or index > self.getLength():raise IndexError("index 非法")# 在单链表中找到插入的位置,即第index-1个结点,并由pre指针指示# 设置i,用于判断是否找到index-1的位置i = 1# 声明pre指针,初始化为指向头结点,用于指示第index-1个结点pre = self.head# 循环,直到pre空,或指向第index-1个结点while pre != None and i !=index:pre = pre.nexti += 1# 创建新节点,在新节点中存储数据元素newNode = Node(data)# 修改指针指向,以实现插入newNode.next = pre.nextpre.next = newNode
删除操作
-
核心思想:
- 循环,找到需要删除的位置,用指针pre指向要删除的前一个结点
- 将pre的next指向被删除结点的后继结点
-
删除方法示意图

-
代码定义方法
def delete(self, index):"""在单链表中任意位置删除元素:param index: 待删除的位置:return: 被删除的结点"""# 判断是否合法if index <= 0 or index > self.getLength():raise IndexError("index 非法")# 判断是否为空if self.isEmpty():raise IndexError("单链表为空,不允许删除")i = 1# 用于遍历单链表,初始化为头结点pre = self.head# 循环,当pre为空或pre为第index-1个结点结束while pre != None and i != index:pre = pre.nexti += 1# 获取被删除的结点delNode = pre.next# 修改指针指向,实现删除pre.next = delNode.nextreturn delNode
合并操作
-
核心思想:
- 通过更改结点的next域来重建新结点之间的线性关系;
- 利用尾插法创建listC单链表,以保证新表有序;
-
合并方法示意图

-
代码定义方法
def merge(self, listB):"""合并操作:param listB: 待合并的单链表B:return: 合并后的单链表listC"""# 声明nodeA和B指针分别指向listA和B当前待合并的结点,分别初始化为listA和B的第一个结点nodeA = self.head.nextnodeB = listB.head.next# 创建listC,沿用listA的头结点listC = LinkedList()listC.head = self.head# 定义tailC作为listC的尾指针,用于合并数据元素,初始化listC的头结点tailC = listC.head# 循环,将较小的结点插入listC中while nodeA != None and nodeB != None:if nodeA.data <= nodeB.data:tailC.next = nodeAtailC = nodeAnodeA =nodeA.nextelse:tailC.next = nodeBtailC = nodeBnodeB = nodeB.next# 如果listA未合并结束,则将listA中剩余的结点链接到listC的表尾if nodeA != None:tailC.next = nodeA# 如果listB未合并结束,则将listB中剩余的结点连接到listC的表尾if nodeB != None:tailC.next = nodeBreturn listC
单链表总代码与调试
- 链式存储结构的优缺点
- 优点:不用考虑单链表的存储空间大小,只要内存足够大,就可以无限延伸;插入或删除方便,只要改变指针域的指向,不用移动大量的结点,效率较顺序表高很多。
- 缺点:逻辑地址与物理地址不一定有直接的映射关系,随机存取单链表中的任一元素的效率较顺序表低。
# 4.单链表的实现
class Node:"""定义节点类型"""def __init__(self, data):# 存储数据元素的数据域self.data = data# 存储指向后继结点位置的指针域self.next = Noneclass LinkedList:"""单链表的定义"""def __init__(self):"""单链表的初始化"""# 声明头结点self.head = Node(None)def isEmpty(self):"""判断单链表是否为空表:return: True or False"""# 如果头结点的next域为空,则返回True,否则返回falsereturn self.head.next == Nonedef getLength(self):"""获取单链表的长度:return: 当前单链表中数据元素的个数"""# 设置length,用来计算长度,初始值为0length = 0# 声明cur指针,用于遍历单链表,初始值为第一个结点cur = self.head.next# 循环,直到cur==Nonewhile cur != None:# 单链表长度加1length += 1# cur指针指向当前结点的后继cur = cur.nextreturn lengthdef display(self):"""展示单链表:return:"""#判断是否为空表if self.isEmpty():print("当前单链表为空表!")return# 遍历单链表,并展示数据元素print("单链表中的数据元素:", end="")cur = self.head.nextwhile cur!= None:print(cur.data, end=",")cur = cur.nextprint()def prepend(self, data):"""头插法:param data: 待插入的元素:return:"""# 创建新结点,在新结点中存储元素newNode = Node(data)# 修改指针指向,实现插入# 将新结点的next指向头结点的后继结点newNode.next = self.head.next# 将头结点的next指向新节点self.head.next = newNodedef append(self, data):"""尾插法:param data: 待插入的元素:return:"""# 创建尾指针,初始化指向头结点rear = self.head# 循环,寻找尾结点while rear.next != None:rear = rear.next# 创建新结点,在新节点中存储元素newNode = Node(data)# 修改指针指向,实现插入# 将尾指针的next指向新结点rear.next = newNodedef getData(self, index):"""按序号查找:param index: 待查找的序号:return: 待查找位置对应的元素值"""# 判断index是否合法if index <= 0 or index > self.getLength():raise IndexError('index 非法')# 声明变量作为计数器,累计当前的结点数。初始为0i = 0# 声明指针cur,用于遍历cur = self.head# 循环扫描while cur != None and i != index:cur = cur.nexti += 1return curdef locate(self, key):"""按内容查找:param key: 待查找的内容:return: 查找内容的位置序号"""# 声明index为索引位置,初始-1index = -1cur = self.head.nexti = 0while cur != None:i += 1if cur.data == key:index = ibreakcur = cur.nextreturn indexdef insert(self, index, data):"""在单链表中任意位置插入元素:param index: 待插入的位置:param data: 待插入的元素:return:"""# 判断index是否合法if index <= 0 or index > self.getLength():raise IndexError("index 非法")# 在单链表中找到插入的位置,即第index-1个结点,并由pre指针指示# 设置i,用于判断是否找到index-1的位置i = 1# 声明pre指针,初始化为指向头结点,用于指示第index-1个结点pre = self.head# 循环,直到pre空,或指向第index-1个结点while pre != None and i !=index:pre = pre.nexti += 1# 创建新节点,在新节点中存储数据元素newNode = Node(data)# 修改指针指向,以实现插入newNode.next = pre.nextpre.next = newNodedef delete(self, index):"""在单链表中任意位置删除元素:param index: 待删除的位置:return: 被删除的结点"""# 判断是否合法if index <= 0 or index > self.getLength():raise IndexError("index 非法")# 判断是否为空if self.isEmpty():raise IndexError("单链表为空,不允许删除")i = 1# 用于遍历单链表,初始化为头结点pre = self.head# 循环,当pre为空或pre为第index-1个结点结束while pre != None and i != index:pre = pre.nexti += 1# 获取被删除的结点delNode = pre.next# 修改指针指向,实现删除pre.next = delNode.nextreturn delNodedef merge(self, listB):"""合并操作:param listB: 待合并的单链表B:return: 合并后的单链表listC"""# 声明nodeA和B指针分别指向listA和B当前待合并的结点,分别初始化为listA和B的第一个结点nodeA = self.head.nextnodeB = listB.head.next# 创建listC,沿用listA的头结点listC = LinkedList()listC.head = self.head# 定义tailC作为listC的尾指针,用于合并数据元素,初始化listC的头结点tailC = listC.head# 循环,将较小的结点插入listC中while nodeA != None and nodeB != None:if nodeA.data <= nodeB.data:tailC.next = nodeAtailC = nodeAnodeA =nodeA.nextelse:tailC.next = nodeBtailC = nodeBnodeB = nodeB.next# 如果listA未合并结束,则将listA中剩余的结点链接到listC的表尾if nodeA != None:tailC.next = nodeA# 如果listB未合并结束,则将listB中剩余的结点连接到listC的表尾if nodeB != None:tailC.next = nodeBreturn listCif __name__ == '__main__':# print('PyCharm')# 1.初始化单链表list = LinkedList()list.display()# 2-0.创建单链表,采用头插入操作创建for i in range(10):list.prepend(ord("A") + i)print("头插法操作结果:", end='')list.display()# 2-1.创建单链表,采用尾插法进行操作list = LinkedList()for i in range(10):list.append(chr(ord("A") + i))print('尾插法操作结果:', end='')list.display()# 3.获取单链表的长度length = list.getLength()print('当前单链表中的长度为:%d' % length)# 4.在表中任意位置插入元素list.insert(2, "T")list.display()# 5-0.按序号查找node = list.getData(2)print('按序号查找的结果:%s' % node.data)# 5-1.按内容查找index = list.locate("B")if index == -1:print("未找到你要的内容位置")else:print("按内容查找的结果:%d" % index)# 6.在单链表中任意位置删除元素node = list.delete(2)print("被删除的数据元素为: %s" % node.data)list.display()# 7.合并操作# 原始元素dataA = (8, 21, 25, 49, 62)dataB = (16, 37, 54, 72, 82, 90)# 创建listAlistA = LinkedList()for i in range(len(dataA)):listA.append(dataA[i])listA.display()# 创建listBlistB = LinkedList()for i in range(len(dataB)):listB.append(dataB[i])listB.display()# 合并成listClistC = listA.merge(listB)listC.display()
相关文章:
【2】数据结构的单链表章
目录标题 单链表的定义单链表的初始化单链表的建立头插法创建尾插法创建 查找操作按序号查找按内容查找 插入操作删除操作合并操作 单链表总代码与调试 单链表的定义 结点(Node)的定义:数据域(data)和指针域ÿ…...
Linux(十一)fork实例练习、文件操作示例及相关面试题目分享
一、fork实例练习 1、思考下面这段代码的打印结果是什么? #include<stdio.h> #include<unistd.h> #include<assert.h> #include<stdlib.h>int main(){int i0;for(;i<2;i){fork();printf("A\n");} exit(0); }所以一共打印6…...
android Fragment使用
在 Android Fragment 中,导入 id(findViewById)并给控件赋值的逻辑通常应该写在 onViewCreated() 方法中,而不是 onCreateView()。 Fragment 生命周期 & 适合的位置 方法作用适合的操作onCreateView()创建并返回 Fragment 的…...
open3d教程 (三)点云的显示
官方文档位置: Visualization - Open3D 0.19.0 documentationhttps://www.open3d.org/docs/release/tutorial/visualization/visualization.html核心方法: o3d.visualization.draw_geometries([几何对象列表]) import open3d as o3dprint("Load …...
根据模板将 Excel 明细数据生成 Txt 文档|邮件合并
在日常办公中,我们常常会遇到需要批量生成文档的任务。以往,若要将 Excel 中的每一条数据都转化为单独的文档,且文档部分内容依据 Excel 数据动态变化,手动操作不仅繁琐,还容易出错。现在,有一种便捷的方法…...
【学Rust写CAD】22 双圆径向渐变的结构体(two_circle_radial_gradient.rs)
源码 //two_circle_radial_gradient.rs //! 定义双圆径向渐变的结构体和相关功能/// 表示一个双圆径向渐变的源 /// /// 该结构体描述了两个圆之间的渐变,支持矩阵变换和颜色查找表优化 #[derive(Debug, Clone, PartialEq)] pub struct TwoCircleRadialGradientSou…...
LVGL Dropdown和Calendar详解
LVGL Dropdown和Calendar详解 一、Dropdown详解创建和初始化设置下拉框选项获取选项获取选中项文本:获取选中项索引:设置选中项: 事件处理其他功能和样式设置设置下拉按钮样式:设置下拉框方向:设置最大高度:…...
AISEO (GEO )中的知识图谱
一、知识图谱在AI SEO中的概念与结构 1. 知识图谱是什么? 定义:知识图谱(Knowledge Graph)是一种以图结构组织的语义网络,由实体(Entity)、**关系(Relation)和属性&…...
Vulnhub-zico2靶机打靶记录
本篇文章旨在为网络安全渗透测试靶机教学。通过阅读本文,读者将能够对渗透Vulnhub系列zico2靶机有一定的了解 一、信息收集阶段 靶机下载地址:https://download.vulnhub.com/zico/zico2.ova 因为靶机为本地部署虚拟机网段,查看dhcp地址池设…...
(041)05-01-自考数据结构(20331)树与二叉树大题总结
实际考试中,计算题约占40%,推理题约占30%,算法设计题约占30%。建议重点练习遍历序列相关的递归分治解法, 知识拓扑 知识点介绍 一、计算题类型与解法 1. 结点数量计算 题型示例: 已知一棵完全二叉树的第6层有8个叶子结点,求该二叉树最多有多少个结点? 解法步骤: 完…...
Python----机器学习(KNN:使用数学方法实现KNN)
一、原理 以下是K最近邻(K-Nearest Neighbors,简称KNN)算法的基本流程,用于对给定点进行分类预测。 1. 获得要预测的点 point_predict 。 2. 计算训练点集 point_set_train 中各点到要预测的点 表 l ist_L2_distance 。 3. 对 poi…...
网络攻防快速入门笔记pwn | 02 栈溢出题型 | 2.2 ret2libc
上一篇:网络攻防快速入门笔记pwn | 02 栈溢出题型 | 2.1 ret2text和ret2shellcode 下一篇:网络攻防快速入门笔记pwn | 02 栈溢出题型 | 2.3 ret2syscall 欢迎关注~ ret2libc 一、 什么是ret2libc(一)ret2lib的概念(…...
Edge浏览器快速开启IE模式
一些老旧的网站,仅支持Internet Explorer(IE)浏览器访问。 然而,出于安全性的考虑,可能会遇到限制IE浏览器使用的情况。 Microsoft Edge浏览器提供了兼容性配置,可以通过IE模式访问这些网站。 以下是两种…...
技术长期主义:用本分思维重构JavaScript逆向知识体系(一)Babel、AST、ES6+、ES5、浏览器环境、Node.js环境的关系和处理流程
基础不牢,地动山摇,逆向越久,越发现基础的重要性,本系列,回顾js逆向基础,让自己的知识体系更加系统化。 以下是 Babel、AST、ES6、ES5、浏览器环境、Node.js环境 的关系和流程的详细说明及图表:…...
Oracle 数据库系统全面详解
Oracle 数据库是全球领先的关系型数据库管理系统(RDBMS),由 Oracle 公司开发。它为企业级应用提供了高性能、高可用性、安全性和可扩展性的数据管理解决方案。 目录 一、Oracle 数据库体系结构 1. 物理存储结构 主要组件: 存储层次: 2. …...
LeetCode 解题思路 29(Hot 100)
解题思路: 映射关系建立:创建一个哈希表存储数字到字母的映射。递归参数: 给定字符串 digits、结果集 result、当前路径 path、当前位置 start。递归过程: 当当前位置 start 等于 digits 长度时,说明已经遍历完 digi…...
使用Python解析PPT文件并生成JSON结构详解
引言 PowerPoint(PPT)文件的自动化处理是办公自动化和数据提取的常见需求。本文将介绍如何通过Python的python-pptx库,将PPT文件的样式、结构、文本内容等信息解析为标准化的JSON格式,为后续的自动化处理、数据迁移或样式复用提供…...
LabVIEW永磁同步电机性能测试系统
开发了一种基于LabVIEW的永磁同步电机(PMSM)性能测试系统的设计及应用。该系统针对新能源汽车使用的电机进行稳态性能测试,解决了传统测试方法成本高、效率低的问题,实现了测试自动化,提高了数据的准确性和客观性。 …...
MTK Camera 照片切视频Systrace拆解分析
和你一起终身学习,这里是程序员Android 经典好文推荐,通过阅读本文,您将收获以下知识点: 一、Systrace 拆解概览二、Systrace 阶段拆解详解 一、Systrace 拆解概览 MTK Camera 照片切换视频trace 拆解(非切换摄像头类) 照片切换视频模块trace…...
某合约任意提取BNB漏洞
1背景描述 合约是一个在满足特定条件时在区块链上执行代码的程序,各方以数字签署合同的方式准许并维护它的其运行。这些代码可以是向朋友汇款、买卖 NFT 虚拟商品等一系列复杂的内容。 存在漏洞的目标合约是一个结合Meme文化病毒式传播与去中心化金融(D…...
Linux_3.2
今天继续学习shell语法 shell类似一个面向过程的语言,要区分好面向过程和面向对象的语言的区别。 循环语句 for循环 for i in a 2 cc doecho $i done #输出a 2 ccfor file in `ls` doecho $file done #输出ls命令的输出for i in $(seq 1 10) doecho $i done #输出1-10,seq…...
插件实现:分别通过winform和WPF界面输入操作CAD——CAD c#二次开发
效果如下图所示: 主程序 using Autodesk.AutoCAD.ApplicationServices; using Autodesk.AutoCAD.DatabaseServices; using Autodesk.AutoCAD.EditorInput; using Autodesk.AutoCAD.Geometry; using Autodesk.AutoCAD.Runtime; using System; using System.Windows…...
【学Rust写CAD】20 平铺模式结构体(spread.rs)
这个 Spread。rs文件定义了渐变超出定义区域时的扩展方式,通常用于处理渐变在边界之外的行为。 源码 //color/spread.rs #[derive(Debug, Clone, Copy)] pub struct Pad; // 空结构体,表示 Pad 模式#[derive(Debug, Clone, Copy)] pub struct Reflect…...
maya调整全局关节显示大小
请按以下步骤操作: 在 Maya 主菜单栏中,找到 Display (显示) 菜单。 在 Display 菜单下,找到 Animation (动画) 子菜单。 在 Animation 子菜单中,点击 Joint Size... (关节大小...)。 这时会弹出一个小窗口或者直接在界面上出现…...
白酒迈入3.0时代,珍酒李渡如何穿越周期高质增长?
当下,白酒行业仍处深度调整期,过往通过渠道拓展、硬广宣传等推动规模扩张、提升市场份额的模式,愈发难以为继。 行业迫切需要构建高质增长新模式,完成增长动能转换。中国酒业协会理事长宋书玉提出,白酒消费亟需进入品…...
HTTP代理:网页加速的隐形引擎
目录 引言:网页加载速度为何至关重要? 一、HTTP代理的核心加速原理 二、四大加速黑科技详解 三、实战场景性能对比 四、代理加速的隐藏代价 五、未来发展趋势 结语:智能代理的选型指南 引言:网页加载速度为何至关重要&#…...
人工智能-LangGraph+ChatUI+DeepSeek API搭建本地智能助手
人工智能-LangGraphChatUIDeepSeek API搭建本地智能助手 0 环境说明1 LangGraph2 Agent Chat UI 0 环境说明 环境项环境说明操作系统Windows11 专业版硬件信息联想拯救者Y9000PcondaAnancondaPython版本3.12NodeJs18.20.0 # 使用conda创建python环境 conda create -n langgra…...
3dmax批量转glb/gltf/fbx/osgb/stl/3ds/dae/obj/skp格式导出转换插件,无需一个个打开max,材质贴图在
3dmax批量转glb/gltf/fbx/osgb/stl/3ds/dae/obj/skp格式导出转换插件,无需一个个打开max,材质贴图在 3dmax批量转glb/gltf/fbx/osgb/stl/3ds/dae/obj/skp格式导出转换插件,无需一个个打开max,材质贴图在...
虚幻5入门
常用操作 运行时,调试相机,按~键,输入ToggleDebugCamera 。进入自由视角 常用节点 gate节点:用于控制该流程通不通,执不执行。Flip Flop节点:反转执行,一次A,一次B。Set Timer by…...
【解决】Edge浏览器硬件加速问题:无法滚动与卡顿的应对方法
Edge浏览器开启硬件加速后无法滚动屏幕,关闭后虽然可以滚动但出现卡顿,可能是由多种原因导致的。以下是一些可能的解决方法: 1. 检查显卡驱动 更新显卡驱动:确保显卡驱动是最新版本。过时的驱动可能会导致硬件加速功能不稳定。回…...
