Python数据结构【四】排序(二)难度:困难

2024-04-20 07:20

本文主要是介绍Python数据结构【四】排序(二)难度:困难,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
    • 一、书接上回
    • 二、快速排序(Quick Sort)
      • 2.1 快速排序思想
      • 2.2 快速排序代码实现
      • 2.3 快速排序复杂度分析
    • 三、堆排序(Heap Sort)
      • 3.1 堆排序思想
      • 3.2 堆排序代码实现
      • 3.3 堆排序复杂度分析
  • 结语

前言

可私聊进一千多人Python全栈交流群(手把手教学,问题解答)

可私聊获取源代码和动画PPT进群可领取Python全栈教程视频 + 多得数不过来的计算机书籍:基础、Web、爬虫、数据分析、可视化、机器学习、深度学习、人工智能、算法、面试题等。

  • 🚀🚀加入我一起学习进步,一个人可以走的很快,一群人才能走的更远!

  • 一、书接上回

    上次学习了冒泡排序,插入排序,选择排序,今天需要学习的是快速排序,堆排序。

    相比于前面三个,这两个可以说是有一点点难度的了。

    二、快速排序(Quick Sort)

    快速排序,突出的就是一个快字,比前面三个时间复杂度少一点,因为快排使用了类似于二分+递归的方法。

    2.1 快速排序思想

    1. 取出一个元素(通常是第一个)

    2. 然后将该元素放入一个位置使得左边的元素都小于该元素,右边的元素都大于该元素

    3. 然后对该元素左右两边进行切割

    4. 然后递归

    5. 结束

    整了个动画来讲解第二步

    在这里插入图片描述

    2.2 快速排序代码实现

    由于第二步功能较多,所以我们可以使用一个函数封装起来。

    找中位数函数(也就是动图的实现)

    def find_mid(left, right, li):"""left: 左指针right: 右指针li: 列表之所以不是0和n-1是为了递归的时候也能使用"""swep = False                        # 代表左边有无空temp = li[left]while left < right:                 # 循环终止条件if swep == False:               # 如果左边有空位if li[right] < temp:        # 判断是否可以交换li[right],li[left] = li[left],li[right]     # 交换swep = True             # 左边无空位else:right -= 1                # 不交换,左边有空位,右指针移动else:                           # 左边无空位if li[left] > temp:         # 判断li[left],li[right] = li[right],li[left]     # 交换swep = False            # 左边有空位else:left += 1               # 无法腾出空位,左指针移动li[left] = temp                     # 找到位置,将元素插入return left                         # 此时左右重合,退出函数
    

    快排递归函数:

    def quick_sort(left, right, li):if left < right:                    # 递归终止条件mid = find_mid(left, right, li)quick_sort(left, mid-1, li)     # 排序左边的quick_sort(mid+1, right, li)    # 排序右边的
    

    测试用例:[5, 4, 2, 8, 3, 9]

    运行结果:

    在这里插入图片描述

    测试用例:[1, 1, 4, 5, 1, 4]

    运行结果:

    在这里插入图片描述

    2.3 快速排序复杂度分析

    时间复杂度:nlogn

    空间复杂度:nlogn (实际上这个是递归的空间复杂度,因为原地排序是不需要空间复杂度的)

    当然这些都是算平均的复杂度,时间上算法的时间空间复杂度都是由最优和最劣的情况的,这里就不一一赘述了

    三、堆排序(Heap Sort)

    堆排序,就是使用堆来进行排序的一种排序方法。

    • 堆——堆是一种特殊的完全二叉树
      • 大根堆——一种完全二叉树,满足任意节点都比孩子节点大
      • 小根堆——一种完全二叉树,满足其任意节点都比孩子节点小

    在这里插入图片描述

    当然树这个数据结构是没有箭头的,我只是想起来已经做出来了就懒得重做了。

    3.1 堆排序思想

    通过堆的向下调整来排序。(左右子树都满足大/小根堆,但是堆本身不满足)

    1. 建立一个堆(需要从小到大排就建立小根堆,需要从大到小就建立大根堆)
    2. 得到堆顶元素(此时堆是大根/小根堆,是有序的)
    3. 将堆顶元素去掉,替换为堆最后的元素,进行堆的向下调整(此时堆顶元素为最大/最小元素)
    4. 重复步骤2,3,直到堆为空(使用递归,判断根节点是否为空)

    由于堆的向下调整比较难,所以只做一个向下调整的演示动画。

    在这里插入图片描述

    如果将上述堆通过列表展示,则列表为:[2, 5, 4, 3, 8, 7]

    从根节点向下,然后每个子树都完全展示出来

    3.2 堆排序代码实现

    向下调整函数:

    def change_down(li, low, high):"""li: 无序数组low: 根节点元素high: 最后一个元素位置"""temp = li[low]                               # 存起来栈顶元素i = low                                      # i指向根节点j = 2 *i + 1                                 # j指向左孩子节点while j <= high:                             # 只要j不为空if j + 1 <= high and li[j+1] > li[j]:    # 如果右孩子节点大j = j + 1                            # j指向右孩子节点  if li[j] > temp:                         # 如果左孩子节点大li[i] = li[j]                        # 将大的放在栈顶i = j                                # 下一层j = 2* i + 1                         # 重新设定左孩子结点else:                                    # 如果栈顶元素是最大的li[i] = temp                         # 将栈顶元素存入栈顶或者某个子树的根节点break                                # 结束循环else:li[i] = temp                             # 将原栈顶元素放在叶子结点
    

    堆排序代码:

    def heap_sort(li):n = len(li)for i in range((n-2)//2, -1, -1):            # 从叶子结点的子树向每个子树的根节点遍历change_down(li, i, n-1)                  # 建堆for i in range(n-1, -1, -1):                 # 挨个取出栈顶元素li[0], li[i] = li[i], li[0]              # 将栈顶元素和最后一个元素进行交换change_down(li, 0, i-1)                  # 向下调整

    测试用例:[2, 4, 8, 7, 3, 5]

    运行截图:

    在这里插入图片描述

    测试用例:[5, 4, 2, 8, 3, 9, 5]

    运行截图:

    在这里插入图片描述

    3.3 堆排序复杂度分析

    时间复杂度:O(nlogn)

    空间复杂度:O(1) 如果使用一个辅助列表,那就是O(n)

    结语

    俗话说得好,光说不练——假把式。一定要多练,多思考。

    想不通就先不想,先去练,多练几次就明白了。

    正如古话说得好,书读百遍其义自见!

这篇关于Python数据结构【四】排序(二)难度:困难的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/919622

相关文章

详解如何通过Python批量转换图片为PDF

《详解如何通过Python批量转换图片为PDF》:本文主要介绍如何基于Python+Tkinter开发的图片批量转PDF工具,可以支持批量添加图片,拖拽等操作,感兴趣的小伙伴可以参考一下... 目录1. 概述2. 功能亮点2.1 主要功能2.2 界面设计3. 使用指南3.1 运行环境3.2 使用步骤4. 核

Python 安装和配置flask, flask_cors的图文教程

《Python安装和配置flask,flask_cors的图文教程》:本文主要介绍Python安装和配置flask,flask_cors的图文教程,本文通过图文并茂的形式给大家介绍的非常详细,... 目录一.python安装:二,配置环境变量,三:检查Python安装和环境变量,四:安装flask和flas

使用Python自建轻量级的HTTP调试工具

《使用Python自建轻量级的HTTP调试工具》这篇文章主要为大家详细介绍了如何使用Python自建一个轻量级的HTTP调试工具,文中的示例代码讲解详细,感兴趣的小伙伴可以参考一下... 目录一、为什么需要自建工具二、核心功能设计三、技术选型四、分步实现五、进阶优化技巧六、使用示例七、性能对比八、扩展方向建

基于Python打造一个可视化FTP服务器

《基于Python打造一个可视化FTP服务器》在日常办公和团队协作中,文件共享是一个不可或缺的需求,所以本文将使用Python+Tkinter+pyftpdlib开发一款可视化FTP服务器,有需要的小... 目录1. 概述2. 功能介绍3. 如何使用4. 代码解析5. 运行效果6.相关源码7. 总结与展望1

使用Python实现一键隐藏屏幕并锁定输入

《使用Python实现一键隐藏屏幕并锁定输入》本文主要介绍了使用Python编写一个一键隐藏屏幕并锁定输入的黑科技程序,能够在指定热键触发后立即遮挡屏幕,并禁止一切键盘鼠标输入,这样就再也不用担心自己... 目录1. 概述2. 功能亮点3.代码实现4.使用方法5. 展示效果6. 代码优化与拓展7. 总结1.

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的

使用Python开发一个简单的本地图片服务器

《使用Python开发一个简单的本地图片服务器》本文介绍了如何结合wxPython构建的图形用户界面GUI和Python内建的Web服务器功能,在本地网络中搭建一个私人的,即开即用的网页相册,文中的示... 目录项目目标核心技术栈代码深度解析完整代码工作流程主要功能与优势潜在改进与思考运行结果总结你是否曾经

Python基础文件操作方法超详细讲解(详解版)

《Python基础文件操作方法超详细讲解(详解版)》文件就是操作系统为用户或应用程序提供的一个读写硬盘的虚拟单位,文件的核心操作就是读和写,:本文主要介绍Python基础文件操作方法超详细讲解的相... 目录一、文件操作1. 文件打开与关闭1.1 打开文件1.2 关闭文件2. 访问模式及说明二、文件读写1.

Python将博客内容html导出为Markdown格式

《Python将博客内容html导出为Markdown格式》Python将博客内容html导出为Markdown格式,通过博客url地址抓取文章,分析并提取出文章标题和内容,将内容构建成html,再转... 目录一、为什么要搞?二、准备如何搞?三、说搞咱就搞!抓取文章提取内容构建html转存markdown

Python获取中国节假日数据记录入JSON文件

《Python获取中国节假日数据记录入JSON文件》项目系统内置的日历应用为了提升用户体验,特别设置了在调休日期显示“休”的UI图标功能,那么问题是这些调休数据从哪里来呢?我尝试一种更为智能的方法:P... 目录节假日数据获取存入jsON文件节假日数据读取封装完整代码项目系统内置的日历应用为了提升用户体验,