当前位置: 首页 > news >正文

【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,否则返回false
        return self.head.next == None
  • 求单链表的长度
    def getLength(self):
        """
        获取单链表的长度
        :return: 当前单链表中数据元素的个数
        """
        # 设置length,用来计算长度,初始值为0
        length = 0
        # 声明cur指针,用于遍历单链表,初始值为第一个结点
        cur = self.head.next
        # 循环,直到cur==None
        while cur != None:
            # 单链表长度加1
            length += 1
            # cur指针指向当前结点的后继
            cur = cur.next
        return length
  • 打印单链表
    def display(self):
        """
        展示单链表
        :return:
        """
        #判断是否为空表
        if self.isEmpty():
            print("当前单链表为空表!")
            return
        # 遍历单链表,并展示数据元素
        print("单链表中的数据元素:", end="")
        cur = self.head.next
        while cur!= None:
            print(cur.data, end=",")
            cur = cur.next
        print()

单链表的建立

头插法创建
  • 将需加入的结点,不断加入到头结点的后面。

  • 核心思想:

    • 将新结点的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 非法')
        # 声明变量作为计数器,累计当前的结点数。初始为0
        i = 0
        # 声明指针cur,用于遍历
        cur = self.head
        # 循环扫描
        while cur != None and i != index:
            cur = cur.next
            i += 1
        return cur
按内容查找
  • 在表中查找与内容值key相同的结点,找到了返回下标,否则返回空序号-1。

  • 代码定义方法

    def locate(self, key):
        """
        按内容查找
        :param key: 待查找的内容
        :return: 查找内容的位置序号
        """
        # 声明index为索引位置,初始-1
        index = -1
        cur = self.head.next
        i = 0
        while cur != None:
            i += 1
            if cur.data == key:
                index = i
                break
            cur = cur.next
        return 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.next
            i += 1
        # 创建新节点,在新节点中存储数据元素
        newNode = Node(data)
        # 修改指针指向,以实现插入
        newNode.next = pre.next
        pre.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.next
            i += 1
        # 获取被删除的结点
        delNode = pre.next
        # 修改指针指向,实现删除
        pre.next = delNode.next
        return delNode

合并操作

  • 核心思想:

    • 通过更改结点的next域来重建新结点之间的线性关系;
    • 利用尾插法创建listC单链表,以保证新表有序;
  • 合并方法示意图
    在这里插入图片描述

  • 代码定义方法

    def merge(self, listB):
        """
        合并操作
        :param listB: 待合并的单链表B
        :return: 合并后的单链表listC
        """
        # 声明nodeA和B指针分别指向listA和B当前待合并的结点,分别初始化为listA和B的第一个结点
        nodeA = self.head.next
        nodeB = 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 = nodeA
                tailC = nodeA
                nodeA =nodeA.next
            else:
                tailC.next = nodeB
                tailC = nodeB
                nodeB = nodeB.next
        # 如果listA未合并结束,则将listA中剩余的结点链接到listC的表尾
        if nodeA != None:
            tailC.next = nodeA
        # 如果listB未合并结束,则将listB中剩余的结点连接到listC的表尾
        if nodeB != None:
            tailC.next = nodeB
        return listC

单链表总代码与调试

  • 链式存储结构的优缺点
    • 优点:不用考虑单链表的存储空间大小,只要内存足够大,就可以无限延伸;插入或删除方便,只要改变指针域的指向,不用移动大量的结点,效率较顺序表高很多。
    • 缺点:逻辑地址与物理地址不一定有直接的映射关系,随机存取单链表中的任一元素的效率较顺序表低。
# 4.单链表的实现
class Node:
    """
    定义节点类型
    """
    def __init__(self, data):
        # 存储数据元素的数据域
        self.data = data
        # 存储指向后继结点位置的指针域
        self.next = None

class LinkedList:
    """
    单链表的定义
    """
    def __init__(self):
        """
        单链表的初始化
        """
        # 声明头结点
        self.head = Node(None)
    def isEmpty(self):
        """
        判断单链表是否为空表
        :return: True or False
        """
        # 如果头结点的next域为空,则返回True,否则返回false
        return self.head.next == None
    def getLength(self):
        """
        获取单链表的长度
        :return: 当前单链表中数据元素的个数
        """
        # 设置length,用来计算长度,初始值为0
        length = 0
        # 声明cur指针,用于遍历单链表,初始值为第一个结点
        cur = self.head.next
        # 循环,直到cur==None
        while cur != None:
            # 单链表长度加1
            length += 1
            # cur指针指向当前结点的后继
            cur = cur.next
        return length
    def display(self):
        """
        展示单链表
        :return:
        """
        #判断是否为空表
        if self.isEmpty():
            print("当前单链表为空表!")
            return
        # 遍历单链表,并展示数据元素
        print("单链表中的数据元素:", end="")
        cur = self.head.next
        while cur!= None:
            print(cur.data, end=",")
            cur = cur.next
        print()

    def prepend(self, data):
        """
        头插法
        :param data: 待插入的元素
        :return:
        """
        # 创建新结点,在新结点中存储元素
        newNode = Node(data)
        # 修改指针指向,实现插入
        # 将新结点的next指向头结点的后继结点
        newNode.next = self.head.next
        # 将头结点的next指向新节点
        self.head.next = newNode
    def append(self, data):
        """
        尾插法
        :param data: 待插入的元素
        :return:
        """
        # 创建尾指针,初始化指向头结点
        rear = self.head
        # 循环,寻找尾结点
        while rear.next != None:
            rear = rear.next
        # 创建新结点,在新节点中存储元素
        newNode =  Node(data)
        # 修改指针指向,实现插入
        # 将尾指针的next指向新结点
        rear.next = newNode
    def getData(self, index):
        """
        按序号查找
        :param index: 待查找的序号
        :return: 待查找位置对应的元素值
        """
        # 判断index是否合法
        if index <= 0 or index > self.getLength():
            raise IndexError('index 非法')
        # 声明变量作为计数器,累计当前的结点数。初始为0
        i = 0
        # 声明指针cur,用于遍历
        cur = self.head
        # 循环扫描
        while cur != None and i != index:
            cur = cur.next
            i += 1
        return cur
    def locate(self, key):
        """
        按内容查找
        :param key: 待查找的内容
        :return: 查找内容的位置序号
        """
        # 声明index为索引位置,初始-1
        index = -1
        cur = self.head.next
        i = 0
        while cur != None:
            i += 1
            if cur.data == key:
                index = i
                break
            cur = cur.next
        return index
    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.next
            i += 1
        # 创建新节点,在新节点中存储数据元素
        newNode = Node(data)
        # 修改指针指向,以实现插入
        newNode.next = pre.next
        pre.next = newNode
    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.next
            i += 1
        # 获取被删除的结点
        delNode = pre.next
        # 修改指针指向,实现删除
        pre.next = delNode.next
        return delNode
    def merge(self, listB):
        """
        合并操作
        :param listB: 待合并的单链表B
        :return: 合并后的单链表listC
        """
        # 声明nodeA和B指针分别指向listA和B当前待合并的结点,分别初始化为listA和B的第一个结点
        nodeA = self.head.next
        nodeB = 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 = nodeA
                tailC = nodeA
                nodeA =nodeA.next
            else:
                tailC.next = nodeB
                tailC = nodeB
                nodeB = nodeB.next
        # 如果listA未合并结束,则将listA中剩余的结点链接到listC的表尾
        if nodeA != None:
            tailC.next = nodeA
        # 如果listB未合并结束,则将listB中剩余的结点连接到listC的表尾
        if nodeB != None:
            tailC.next = nodeB
        return listC

if __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)
    # 创建listA
    listA = LinkedList()
    for i in range(len(dataA)):
        listA.append(dataA[i])
    listA.display()
    # 创建listB
    listB = LinkedList()
    for i in range(len(dataB)):
        listB.append(dataB[i])
    listB.display()
    # 合并成listC
    listC = listA.merge(listB)
    listC.display()

相关文章:

  • 【Python】Browser-Use:让 AI 替你掌控浏览器,开启智能自动化新时代!
  • vue3中ref、reactive的使用示例
  • 测试脚本代码质量规范:从可读性到模块化设计的深度解析
  • 数据库 第一章 MySQL(3)
  • LeetCode算法题(Go语言实现)_23
  • 9对象树(3)
  • 组合数学——排列与组合
  • 24.3 CogView3多模态生成实战:从API调优到1024高清图像生成全解析
  • 使用 Provider 和 GetX 实现 Flutter 局部刷新的几个示例
  • oracle 分组函数
  • STM32单片机入门学习——第3-4节: [2-1、2]软件安装和新建工程
  • Netty——心跳监测机制
  • 【Java 优选算法】哈希表
  • 基于springboot+vue的动漫交流与推荐平台
  • Zookeeper经典应用场景实战
  • 【GPT入门】第29课 对比不同的langchain ReAct代理的思考过程
  • C嘎嘎类里面的额函数
  • 一文详细讲解Python(详细版一篇学会Python基础和网络安全)
  • 安卓Q以下自定义文字的字体
  • LVGL Dropdown和Calendar详解
  • 公元1059年:宋朝人为什么这么爱杜甫?
  • 中国—东盟经贸部长特别会议就支持多边贸易体制等达成重要共识
  • 黄仁勋的新逻辑:从“卖铲人”到“全球AI基建运营商”
  • 贵州茅台:支持工作餐不上酒的规定,请投资者相信茅台创新和自我调节能力
  • 欧洲观察室|欧盟对华战略或在中欧建交50年时“低开高走”
  • 河南通报部分未检疫生猪流入:立案查处,涉案猪肉被封存