news 2026/5/1 5:42:49

python--数据结构--链表

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
python--数据结构--链表

最近会更新很多内容,感兴趣的友友支持一下吧!!

一、链表介绍

概述:

属于线性结构, 即: 每个节点都有1个父节点(前驱节点) 和 1个子节点(后继节点)

链表可以看做是 用链条(一根绳) 把节点连接起来的 一种结构.

节点介绍(此处以 单链表举例):

由 元素域(数值域) 和 地址域(链接域)组成, 其中 数值域存储的是 数值, 地址域存储的是 下个节点的地址.

链表根据节点不同, 分类如下:

单向链表:

每个节点由 1个数值域和1个地址域组成, 且每个节点的地址域指向下个节点的地址. 最后1个节点的地址域为: None

单向循环链表:

每个节点由 1个数值域和1个地址域组成, 且每个节点的地址域指向下个节点的地址. 最后1个节点的地址域为: 第1个节点的地址

双向链表:

每个节点由 1个数值域和2个地址域组成, 且每个节点的地址域分别指向前1个节点的地址 和 后1个节点的地址.

第1个节点的(前)地址域为: None, 最后1个节点的(后)地址域为: None

双向循环链表:

每个节点由 1个数值域和2个地址域组成, 且每个节点的地址域分别指向前1个节点的地址 和 后1个节点的地址.

第1个节点的(前)地址域为: 最后1个节点的地址, 最后1个节点的(后)地址域为: 第1个节点的地址.

二、案例:

自定义代码, 模拟(单向)链表.

分析:

节点类: SingleNode

属性:

item 表示: 数值域

next 表示: 地址域, 即: 指向下个节点的地址的.

链表类: SingleLinkedList

属性:

head 表示: 头结点, 即: 默认指向链表第1个节点的 地址.

函数:

is_empty(self) 链表是否为空

length(self) 链表长度

travel(self. ) 遍历整个链表

add(self, item) 链表头部添加元素

append(self, item) 链表尾部添加元素

insert(self, pos, item) 指定位置添加元素

remove(self, item) 删除节点

search(self, item) 查找节点是否存在

示例代码:

# 1. 定义SingleNode, 表示: 节点类. class SingleNode(object): # 1.1 初始化节点的属性. def __init__(self, item): # 1.2 接收由用户传入的: 数值, 放到数值域中. self.item = item # 1.3 地址域默认为: None self.next = None # 2. 定义SingleLinkedList, 表示: 链表类. class SingleLinkedList(object): # 2.1 初始化链表的属性. def __init__(self, head=None): # 2.2 默认: 头结点为: None self.head = head # 2.2 is_empty(self) 链表是否为空 def is_empty(self): # 判断依据: 头结点head是否为None # 写法1: if方式 # if self.head is None: # return True # 链表为空 # else: # return False # 链表不为空 # 写法2: 三元表达式 # return True if self.head is None else False # 写法3: 直接返回即可. # return self.head == None return self.head is None # 2.3 length(self) 链表长度 def length(self): # 1. 定义变量count, 记录: 链表长度. count = 0 # 2. 定义变量cur, 表示: 当前的节点, 初值为: 头结点. cur = self.head # 3. 判断当前节点是否为空, 不为空就一直遍历. while cur is not None: # 4. 走到这里, 说明当前节点不为空, 就: 计数器 + 1, 然后设置当前节点为它的下个节点. count += 1 cur = cur.next # 5. 返回结果(链表的长度). return count # 2.4 travel(self ) 遍历整个链表 def travel(self): # 1. 定义变量cur, 表示: 当前的节点, 初值为: 头结点. cur = self.head # 2. 判断当前节点是否为空, 不为空就一直遍历. while cur is not None: # 3. 走到这里, 说明当前节点不为空. print(cur.item) # 打印当前节点的 数值域 cur = cur.next # 设置当前节点为它的下个节点. # 2.5 add(self, item) 链表头部添加元素 def add(self, item): # 1. 把传入的item(元素值)封装成: 节点 new_node = SingleNode(item) # 2. 设置新节点的 地址域为: 头结点. new_node.next = self.head # 3. 设置新节点为: 新的头结点. self.head = new_node # 2.6 append(self, item) 链表尾部添加元素 def append(self, item): # 1.把要添加的元素值封装成: 节点. new_node = SingleNode(item) # 2. 判断当前链表是否为空. if self.is_empty(): # 3. 走到这里, 说明链表为空, 直接设置新节点为: 头结点即可. self.head = new_node else: # 4. 走到这里, 说明链表不为空, 需找到最后1个节点. cur = self.head # 设置当前节点为: 头结点. # 5. 只要当前节点的地址域不为空, 就说明还有下个节点, 就一直遍历即可. while cur.next is not None: # 当前节点不为空, 继续往下找. cur = cur.next # 6. 走到这里, 说明cur的下个节点为空, 即: cur已经为最后1个节点了. # 设置最后1个节点(cur)的地址域为: 新的节点. cur.next = new_node # 2.7 insert(self, pos, item) 指定位置添加元素 def insert(self, pos, item): # 1. 判断pos(要插入的位置)是否合法. if pos <= 0: # 往头部添加. self.add(item) elif pos >= self.length(): # 往尾部添加 self.append(item) else: # 2. 走到这里, 说明pos是合法位置, 即: 链表的中间位置. # 解题核心: 找到要插入的位置的前边那个节点, 即: pos - 1位置的那个节点. # 3. 定义变量cur, 表示: 要插入的位置前 那个节点. cur = self.head # 4. 定义变量count, 表示: 要插入的位置前 那个节点的 索引. count = 0 # 5. 只要cur不是要插入节点 前一个节点, 就一直遍历. while count < pos - 1: # 如果不减1, 拿的是插入后. pos位置后的哪个元素. # 走到这里, 说明cur不是要找的元素, 就继续往后找. cur = cur.next # 计数器 要 + 1 count += 1 # 6. 走到这里, 说明cur就是 要插入位置前的那个节点. # 先设置 新节点的地址域为: cur节点的地址域 new_node = SingleNode(item) new_node.next = cur.next # 再设置 cur节点的地址域为: 新节点 cur.next = new_node # 2.8 remove(self, item) 删除节点 def remove(self, item): # 1. 定义变量, 表示: 要被删除的元素. cur = self.head # 从头结点往后找. # 2. 定义变量, 表示: 要被删除的节点的 前1个节点. pre = None # 初值为None # 3. 具体的判断动作, 只要当前节点不为空, 就一直遍历. while cur is not None: # 4. 判断当前节点是否是要被删除的节点. if cur.item == item: # 走这里, 说明cur就是要被删除的节点. # 5. 判断cur是否是头结点, 如果是头结点, 直接设置它的地址域 为 新的头结点即可. if cur == self.head: self.head = cur.next else: # 6. 走这里, 说明cur不是头结点, 设置它的前1个节点的地址域 = cur的地址域即可. pre.next = cur.next # 7. 删完以后, 记得: break break else: # 8. 走这里, 说明cur不是要被删除的节点, 我们继续往下找. pre = cur # 当前节点已经是: 要被删除的节点的 前1个节点了 cur = cur.next # 当前节点变更为: 它的下个节点. # 2.9 search(self, item) 查找节点是否存在 def search(self, item): # 1. 定义变量cur, 表示: 当前节点, 默认从头结点开始. cur = self.head # 2. 判断当前节点是否为空, 不为空就一直遍历. while cur is not None: # 3. 判断当前节点的 元素域 是否和 要查找的元素值相同. if cur.item == item: # 4. 走这里, 找到了, return True即可. return True # 5. 走到这里, 说明当前节点不是我们要的, 继续往下找. cur = cur.next # 6. 走到这里, 说明 没找到, return False即可. return False # 3. 在main方法中测试 if __name__ == '__main__': # 3.1 测试 节点类. node1 = SingleNode('乔峰') print(node1) # 地址: <__main__.SingleNode object at 0x0000015D0D48C5C0> # 3.2 打印该节点的属性 print(node1.item) # 数值域, 乔峰 print(node1.next) # 地址域, None print('-' * 21) # 3.2 测试 链表类 linkedlist1 = SingleLinkedList() # 空链表. print(linkedlist1.head) # 打印链表的: 头结点 print('-' * 21) linkedlist2 = SingleLinkedList(node1) # 打印链表的: 头结点, 这里是: node1节点, <__main__.SingleNode object at 0x0000015D0D48C5C0> print(linkedlist2.head) print(linkedlist2.head.item) # 打印 头结点的 数值域, 乔峰 print(linkedlist2.head.next) # 打印 头结点的 地址域, None print('-' * 21) # 3.3 测试链表的方法: add() linkedlist2.add('虚竹') linkedlist2.add('段誉') # 3.4 测试链表的方法: append() linkedlist2.append('王语嫣') linkedlist2.append('李清露') # 3.5 测试链表的方法: insert() linkedlist2.insert(-2, '鸠摩智') linkedlist2.insert(20, '慕容复') linkedlist2.insert(3, '丁春秋') # 3.6 测试链表的方法: remove() linkedlist2.remove('乔峰') linkedlist2.remove('鸠摩智') # 3.7 测试链表的方法: search() print(linkedlist2.search('慕容复')) print(linkedlist2.search('鸠摩智')) print('#' * 21) # 3.8 测试链表的方法: is_empty() print(linkedlist1.is_empty()) print(linkedlist2.is_empty()) # 3.9 测试链表的方法: length() len1 = linkedlist1.length() len2 = linkedlist2.length() print(f'len1: {len1}') # 0 print(f'len2: {len2}') # 1 print('-' * 21) # 3.10 测试链表的方法: travel() linkedlist2.travel() # 鸠摩智, 段誉, 虚竹, 丁春秋, 乔峰, 王语嫣, 李清露, 慕容复

运行结果:

<__main__.SingleNode object at 0x00000125ED554FE0>
乔峰
None
---------------------
None
---------------------
<__main__.SingleNode object at 0x00000125ED554FE0>
乔峰
None
---------------------
True
False
#####################
True
False
len1: 0
len2: 6
---------------------
段誉
虚竹
丁春秋
王语嫣
李清露
慕容复

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/24 21:15:47

Python---多线程相关内容

最近会更新很多内容,感兴趣的友友点个关注,支持一下博主吧! 一、线程基本内容 概述: 线程是CPU调度资源的最基本单位, 进程是CPU分配资源的基本单位. 进程 = 可执行程序, 文件. 即: *.exe = 进程, 微信, QQ都是进程. 线程 = 进程的执行路径, 执行单元. 微信这个进程, 可以…

作者头像 李华
网站建设 2026/4/19 14:18:58

吐血推荐8个AI论文平台,助你轻松搞定本科毕业论文!

吐血推荐8个AI论文平台&#xff0c;助你轻松搞定本科毕业论文&#xff01; AI 工具助力论文写作&#xff0c;轻松应对学术挑战 对于正在撰写本科毕业论文的同学们来说&#xff0c;时间紧、任务重是常态。从选题到开题&#xff0c;再到撰写初稿和反复修改&#xff0c;每一个环节…

作者头像 李华
网站建设 2026/4/21 19:42:18

20260106_165519_大模型中的MCP、RAG、Agent定义及关系

MCP、RAG、Agent 是 LLM 应用架构的三大核心组件。 1、MCP&#xff08;Model Control Plane&#xff0c;模型控制平面&#xff09; 定义&#xff1a;管理 LLM 全生命周期的 “调度中枢”&#xff0c;负责模型选型、部署、负载均衡、版本控制、权限管理等。核心价值&#xff1…

作者头像 李华
网站建设 2026/4/25 15:34:10

深度学习毕设项目:基于python深度学习的手势识别数字

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

作者头像 李华
网站建设 2026/4/16 5:46:50

拒绝繁琐配置,推荐一款简洁易用的知识管理工具sward

中国区Confluence Server即将停止销售&#xff0c;今天给大家推荐一款国产开源免费的开源平替方案 - sward。 1、产品简介 sward是一款国产的且免费开源的知识管理工具&#xff0c;包含知识库管理、目录管理、文档管理、markdown管理、文档的评审与版本维护等模块。产品页面简…

作者头像 李华
网站建设 2026/4/29 4:51:56

vscode出现中文乱码?

第一次用vscode打开代码会出现中文注释因识别不了而乱码的情况&#xff0c;类似这样printf(“张三\n”);1.单击软件左下角设置&#xff08;类似齿轮&#xff09;的按钮在出现的搜索框里输入&#xff1a;Files.autoGuessEncoding&#xff0c;按回车键确定。关闭vscode,然后重新打…

作者头像 李华