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月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
62 4
|
7天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
68 5
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
113 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
53 0
|
3月前
|
存储
[数据结构] -- 单链表
[数据结构] -- 单链表
29 1
|
2月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
78 0
|
3月前
|
存储
[数据结构] -- 双向循环链表
[数据结构] -- 双向循环链表
28 0