python 数据结构基础一:线性表单链表的实现

简介: 需要理解python的类、实例、赋值原理(其实就是地址的引用)等概念### 二、总体工作先定义一个链表结点类(LNode),用于生成链表结点。然后定义一个单链表对象类(LList),用于存储链表结点、操作结点数据。### 三、实现

@TOC

一、前提

需要理解python的类、实例、赋值原理(其实就是地址的引用)等概念

二、总体工作

先定义一个链表结点类(LNode),用于生成链表结点。然后定义一个单链表对象类(LList),用于存储链表结点、操作结点数据。

三、实现

1.定义链表结点类
class LNode:
    """
    链表结点类
    """
    def __init__(self, elem, next_=None):
        self.elem = elem
        self.next = next_
2.测试结点类对象的使用
# 创建一个链表结点类对象,并赋值给lis
lis = LNode(1)

# 将lis赋值给p(必须,详细参看图片,否则没办法构成链表)
p = lis

# 循环添加结点
for i in range(2, 10):
    p.next = LNode(i)
    p = p.next

# 将lis赋值给q(非必须,只是为了好看)
q = lis
print(lis)
# 循环输出结点
while q:
    print(q.elem)
    q = q.next
print(lis)
3.上面测试原理图(根据赋值原理)
graph RL
B[elem2 next] --> C[elem1 next]
C --> D[lis]
C-->E[p]
C -->F[q]

似乎不好画,我还是口述吧。创建一个链表的结点对象elem1,赋值给lis(相当于引用了elem1的地址)。又将lis赋值给p(相当于lis直接把elem1的地址给了p,及直接引用了elem1的地址)。将新创建的结点elem2地址给elem1结点对象的next属性,再把elem1的next属性赋值给p(其实就是将elem2的地址给p引用),以此类推到都链接完。循环输出结点及直接把结点elem1地址给q(或者直接用lis也行),然后输出当前的元素,并把当前元素next属性的elem2的地址给q即可(就是直接赋值喽),以此类推输出完。
其实此理解原理图反过来就是我们在讲数据结构的书上所画的示意图了。

4.定义单链表对象类(包含一个异常类)
# 异常类
class LinkedListUnderflow(ValueError):
    pass


class LList:
    """
    单链表对象类
    """
    def __init__(self):
        self._head = None

    def is_empty(self):
        """
        判断表是否为空
        :return: True:为空,False:不为空
        """
        return self._head is None

    def prepend(self, elem):
        """
        从首端插入元素
        :param elem:
        :return:
        """
        self._head = LNode(elem=elem, next_=self._head)

    def pop(self):
        """
        删除表头结点,并返回这个结点的数据
        :return: 结点的数据
        """
        if self._head is None:
            raise LinkedListUnderflow('in pop')
        e = self._head.elem
        self._head = self._head.next
        return e

    def append(self, elem):
        """
        在链表最后插入元素
        :return:
        """
        if self._head is None:
            self._head = LNode(elem=elem)
            return
        p = self._head
        while p.next is not None:
            p = p.next
        p.next = LNode(elem=elem)

    def pop_last(self):
        """
        将链表最后的元素删除并返回
        :return:
        """
        # 空表
        if self._head is None:
            raise LinkedListUnderflow('in pop_last')
        p = self._head

        # 表中只有一个元素
        if p.next is None:
            e = p.elem
            self._head = None
            return e

        # 循环直到p.next是最后结点
        while p.next.next is not None:
            p = p.next
        e = p.next.elem
        p.next = None
        return e

    def find(self, pred):
        """
        返回第一个符合谓词的表元素
        :param pred:
        :return:
        """
        p = self._head
        while p is not None:
            if pred(p.elem):
                return p.elem
            p = p.next

    def printall(self):
        """
        输出表中的元素情况
        :return:
        """
        p = self._head
        while p is not None:
            print(p.elem, end='')
            if p.next is not None:
                print(', ', end='')
            p = p.next
        print('')

上面类的定义及方法函数都是基于理解2中的测试结点类对象的使用所写的,其中类中的数据即1中的结点类对象。但链表初始时应该为空,所以在init初始化的时候设置为空。在一些操作中有使用创建结点类对象。

5.测试单链表对象类
# 使用链表,定义链表对象
mlist = LList()

# 判断是否为空表
print(mlist.is_empty())

# 从表的前端插入元素,并查看元素状态
for i in range(10):
    mlist.prepend(i)
mlist.printall()

# 从表的后端插入元素, 并查看元素状态
for i in range(11, 20):
    mlist.append(i)
mlist.printall()

# 从表的前端删除元素,并查看元素状态
pre_node = mlist.pop()
print(pre_node)
mlist.printall()

# 从表的后端删除元素,并查看元素状态
last_node = mlist.pop_last()
print(last_node)
mlist.printall()

# 定义一个谓词函数
def betw(elem):
    if elem in (6, 10, 21):
        return True
    else:
        return False
# 使用这个谓词函数查找对应的数据
findNode = mlist.find(betw)
print(findNode)

# 判空
print(mlist.is_empty())
目录
相关文章
|
2月前
|
存储 缓存 监控
局域网屏幕监控系统中的Python数据结构与算法实现
局域网屏幕监控系统用于实时捕获和监控局域网内多台设备的屏幕内容。本文介绍了一种基于Python双端队列(Deque)实现的滑动窗口数据缓存机制,以处理连续的屏幕帧数据流。通过固定长度的窗口,高效增删数据,确保低延迟显示和存储。该算法适用于数据压缩、异常检测等场景,保证系统在高负载下稳定运行。 本文转载自:https://www.vipshare.com
132 66
|
3月前
|
存储 开发者 索引
Python 中常见的数据结构
这些数据结构各有特点和适用场景,在不同的编程任务中发挥着重要作用。开发者需要根据具体需求选择合适的数据结构,以提高程序的效率和性能
151 59
|
3月前
|
存储 索引 Python
Python编程数据结构的深入理解
深入理解 Python 中的数据结构是提高编程能力的重要途径。通过合理选择和使用数据结构,可以提高程序的效率和质量
171 59
|
3月前
|
存储 开发者 Python
Python 中的数据结构与其他编程语言数据结构的区别
不同编程语言都有其设计理念和应用场景,开发者需要根据具体需求和语言特点来选择合适的数据结构
118 55
|
1月前
|
存储 算法 测试技术
【C++数据结构——线性表】求集合的并、交和差运算(头歌实践教学平台习题)【合集】
本任务要求编写程序求两个集合的并集、交集和差集。主要内容包括: 1. **单链表表示集合**:使用单链表存储集合元素,确保元素唯一且无序。 2. **求并集**:遍历两个集合,将所有不同元素加入新链表。 3. **求交集**:遍历集合A,检查元素是否在集合B中存在,若存在则加入结果链表。 4. **求差集**:遍历集合A,检查元素是否不在集合B中,若满足条件则加入结果链表。 通过C++代码实现上述操作,并提供测试用例验证结果。测试输入为两个集合的元素,输出为有序集合A、B,以及它们的并集、交集和差集。 示例测试输入: ``` a c e f a b d e h i ``` 预期输出:
42 7
|
1月前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
39 5
|
1月前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】顺序表的基本运算(头歌实践教学平台习题)【合集】
本文档介绍了线性表的基本运算任务,涵盖顺序表和链表的初始化、销毁、判定是否为空、求长度、输出、查找元素、插入和删除元素等内容。通过C++代码示例详细展示了每一步骤的具体实现方法,并提供了测试说明和通关代码。 主要内容包括: - **任务描述**:实现顺序表的基本运算。 - **相关知识**:介绍线性表的基本概念及操作,如初始化、销毁、判定是否为空表等。 - **具体操作**:详述顺序表和链表的初始化、求长度、输出、查找、插入和删除元素的方法,并附有代码示例。 - **测试说明**:提供测试输入和预期输出,确保代码正确性。 - **通关代码**:给出完整的C++代码实现,帮助完成任务。 文档
41 5
|
2月前
|
存储 运维 监控
探索局域网电脑监控软件:Python算法与数据结构的巧妙结合
在数字化时代,局域网电脑监控软件成为企业管理和IT运维的重要工具,确保数据安全和网络稳定。本文探讨其背后的关键技术——Python中的算法与数据结构,如字典用于高效存储设备信息,以及数据收集、异常检测和聚合算法提升监控效率。通过Python代码示例,展示了如何实现基本监控功能,帮助读者理解其工作原理并激发技术兴趣。
68 20
|
3月前
|
存储 算法 搜索推荐
Python 中数据结构和算法的关系
数据结构是算法的载体,算法是对数据结构的操作和运用。它们共同构成了计算机程序的核心,对于提高程序的质量和性能具有至关重要的作用
111 33
|
1月前
|
Python
探索 Python 中链表的实现:从基础到高级
链表是一种由节点组成的基础数据结构,每个节点包含数据和指向下一个节点的引用。本文通过Python类实现单向链表,详细介绍了创建、插入、删除节点等操作,并提供示例代码帮助理解。链表在处理动态数据时具有高效性,适用于大量数据变动的场景。文章为初学者提供了全面的入门指南,助你掌握链表的核心概念与应用。