「数据结构」C语言排序方法——堆排序详解
yuyutoo 2024-10-23 16:43 2 浏览 0 评论
堆是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。
对于堆的操作通常需要以下3个步骤:
最大堆调整(Max Heapify):将堆的末端子节点作调整,使得子节点永远小于父节点
创建最大堆(Build Max Heap):将堆中的所有数据重新排序
堆排序(HeapSort):移除位在第一个数据的根节点,并做最大堆调整的递归运算。
C代码实现
代码看起来比较抽象,将代码运行时数据交换的过程打印出来,然后结合二叉树的图形来分析,就会比较好理解了。 代码运行过程中数据交换过程如下:
为了方便观看这里使用二叉树图形生成软件,通过二叉树图形来观察数据交换过程。 二叉树图形生成使用的是一个在线生成网站:mshang.ca/syntree 后面所有的二叉树图形都使用此软件生成。
代码一开始首先打印出原始数据
数组元素 0 8 1 5 4 3 7 9 2 6 将这10个数据在网站上使用公式生成二叉树的图表,软件界面如下:
首先将数组从上至下按顺序排列,转换成二叉树。
公式: 0[8 [5 9[2]][4[6]]] [1[3] [7 ]]]
生成二叉树图表:
转换成二叉树之后,需要让这个无序堆变成最大堆,也就是每个堆都实现父节点的值大于任何一个子节点值。 从最后一个堆开始,依次比较父节点和子节点的值,如果子节点值大于父节点值,就需要交换。
创建最大堆: 6为最后一个子节点,所以从6开始依次和父节点比较,如果子节点大于父节点,就需要交换子节点和父节点的位置。 从设备树图形中可以看出,子节点6大于父节点4,所以需要交换子节点的父节点的位置。
公式:0[8 [5 9[2]][6[4]]] [1[3] [7 ]]]
交换 6 和4
6交换完成后,接下来依次向前比较其他子节点,6前面的节点是2,2小于父节点5,继续向前查找子节点9,子节点9大于父节点5,所以就交换9和5的位置。
公式:0[8 [9 5[2]][6[4]]] [1[3] [7 ]]]
交换9和5
接下来继续向前查找,发现子节点7大于父节点1,继续交换。
公式:0[8 [9 5[2]][6[4]]] [7[3] [1 ]]]
交换7和1
继续向前查找发现子节点9大于父节点8,交换位置。
公式:0[9 [8 5[2]][6[4]]] [7[3] [1 ]]]
交换9和8
继续比较其他节点
公式:9[0 [8 5[2]][6[4]]] [7[3] [1 ]]]
交换9和0
公式:9[8 [0 5[2]][6[4]]] [7[3] [1 ]]]
交换8和0
公式:9[8 [5 0[2]][6[4]]] [7[3] [1 ]]]
交换5和0
此时最大堆已经创建完成,此时根节点的数字9就是数组中的最大值。
代码中打印出来的数据顺序和通过二叉树图形分析出来的顺序完全一样。 此时最大堆调整已经完成了。接下来就需要开始堆排序,依次将根节点的数据存放到最后一个节点,形成一个有序堆。
堆排序(最大堆调整)
首先交换数组中第一个元素,和排序好的前一个元素位置。 此时数组中的第一个元素是9,排序完成之后最后一个元素是4,交换9和4.
公式:4[8 [5 0[2]][6[9]]] [7[3] [1 ]]]
交换9和4
交换完成后,此时最大值9所在的底部位置就成为了有序区,有序区之后就不会参与任何对比。 接下来继续调整父节点和子节点,确保父节点要大于子节点。
公式:8[4 [5 0[2]][6[9]]] [7[3] [1 ]]]
交换8和4
公式:8[6 [5 0[2]][4[9]]] [7[3] [1 ]]]
交换6和4
此时数字8称为了根节点,是目前无序区中的最大值,将8和底部区的2交换,将当前最大值8放到有序区中。
公式:2[6 [5 0[8]][4[9]]] [7[3] [1 ]]]
交换8和2
此时8已经存放到了有序区中,此后就不参与任何交换了。 此时2变为根节点后,需要再重新调整一次节点,确保父节点大于子节点。
公式:7[6 [5 0[8]][4[9]]] [2[3] [1 ]]]
交换7和2
公式:7[6 [5 0[8]][4[9]]] [3[2] [1 ]]]
交换3和2
此时数字7变为根节点,是无序区间的最大值。需要将根节点的数字移动到有序区中。
将根节点7和0交换位置。
公式:0[6 [5 7[8]][4[9]]] [3[2] [1 ]]]
交换7和0
接下来重新调整节点 公式:6[0 [5 7[8]][4[9]]] [3[2] [1 ]]]
交换6和0
公式:6[5 [0 7[8]][4[9]]] [3[2] [1 ]]]
交换5和0
此时6变为了根节点,是无序区间的最大值。
将根节点和有序区间的前一个数字交换,也就是1和6需要交换。
公式:1[5 [0 7[8]][4[9]]] [3[2] [6 ]]]
交换6和1
接下来重新调节一次节点
公式:5[1 [0 7[8]][4[9]]] [3[2] [6 ]]]
交换5和1
公式:5[4 [0 7[8]][1[9]]] [3[2] [6 ]]]
交换4和1
此时5变成的根节点,需要将5移动到有序队列中去。
接下来需要交换根节点5和无序节点2的位置
公式:2[4 [0 7[8]][1[9]]] [3[5] [6 ]]]
交换5和2
重新调整节点位置
公式:4[2 [0 7[8]][1[9]]] [3[5] [6 ]]]
交换4和2
此时4是无序列表中的最大值,需要交换4和1的位置
公式:1[2 [0 7[8]][4[9]]] [3[5] [6 ]]]
交换4和1
重新调整节点位置
公式:9[2 [0 6[7]][3[8]]] [1[4] [5 ]]]
交换3和1
此时3为无序列表中最大值,需要交换3和0的位置。
公式:0[2 [3 7[8]][4[9]]] [1[5] [6 ]]]
交换3和0
交换完成后重新调整节点位置 公式:9[0 [2 6[7]][3[8]]] [1[4] [5 ]]]
交换2和0
此时2变成了无序列表中最大值,1为有序列表的前一个值,交换2和1的位置。
公式:1[0 [3 7[8]][4[9]]] [2[5] [6 ]]]
交换2和1
此时1是根节点,无序列表中就剩0一个数字了,交换1和0。
公式:0[1 [3 7[8]][4[9]]] [2[5] [6 ]]]
交换1和0
这是0变成了根节点,而其他的所有数字都在有序列表中,无序列表中已经没有数字了,此时说明排序完成。
接下来测试一下最坏情况下数字移动情况
在测试一下最好情况下数字移动情况
可以看出最好、最坏、一般情况下数据移动的次数和方法基本都差不多。
接下来随机生成10000个数据,看看排序需要多长时间。
最后用一张动图来演示堆排序的过程
写在最后
另外,对现在我们的大多数朋友来说还是学编程技术最重要!栽一棵树最好的时间是十年前,其次是现在。对于准备学习编程的小伙伴,如果你想更好地提升你的编程核心能力(内功)不妨从现在开始!
编程学习书籍分享:
编程学习视频分享:
整理分享(多年学习的源码、项目实战视频、项目笔记,基础入门教程)
欢迎转行和学习编程的伙伴,利用更多的资料学习成长比自己琢磨更快哦!
想学习C/C++编程,或者对编程感兴趣的话可以【私信】笔者领取免费学习资料哟~
相关推荐
- jQuery VS AngularJS 你更钟爱哪个?
-
在这一次的Web开发教程中,我会尽力解答有关于jQuery和AngularJS的两个非常常见的问题,即jQuery和AngularJS之间的区别是什么?也就是说jQueryVSAngularJS?...
- Jquery实时校验,指定长度的「负小数」,小数位未满末尾补0
-
在可以输入【负小数】的输入框获取到焦点时,移除千位分隔符,在输入数据时,实时校验输入内容是否正确,失去焦点后,添加千位分隔符格式化数字。同时小数位未满时末尾补0。HTML代码...
- 如何在pbootCMS前台调用自定义表单?pbootCMS自定义调用代码示例
-
要在pbootCMS前台调用自定义表单,您需要在后台创建表单并为其添加字段,然后在前台模板文件中添加相关代码,如提交按钮和表单验证代码。您还可以自定义表单数据的存储位置、添加文件上传字段、日期选择器、...
- 编程技巧:Jquery实时验证,指定长度的「负小数」
-
为了保障【负小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【负小数】的方法。HTML代码<inputtype="text"class="forc...
- 一篇文章带你用jquery mobile设计颜色拾取器
-
【一、项目背景】现实生活中,我们经常会遇到配色的问题,这个时候去百度一下RGB表。而RGB表只提供相对于的颜色的RGB值而没有可以验证的模块。我们可以通过jquerymobile去设计颜色的拾取器...
- 编程技巧:Jquery实时验证,指定长度的「正小数」
-
为了保障【正小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【正小数】的方法。HTML做成方法<inputtype="text"class="fo...
- jquery.validate检查数组全部验证
-
问题:html中有多个name[],每个参数都要进行验证是否为空,这个时候直接用required:true话,不能全部验证,只要这个数组中有一个有值就可以通过的。解决方法使用addmethod...
- Vue进阶(幺叁肆):npm查看包版本信息
-
第一种方式npmviewjqueryversions这种方式可以查看npm服务器上所有的...
- layui中使用lay-verify进行条件校验
-
一、layui的校验很简单,主要有以下步骤:1.在form表单内加上class="layui-form"2.在提交按钮上加上lay-submit3.在想要校验的标签,加上lay-...
- jQuery是什么?如何使用? jquery是什么功能组件
-
jQuery于2006年1月由JohnResig在BarCampNYC首次发布。它目前由TimmyWilson领导,并由一组开发人员维护。jQuery是一个JavaScript库,它简化了客户...
- django框架的表单form的理解和用法-9
-
表单呈现...
- jquery对上传文件的检测判断 jquery实现文件上传
-
总体思路:在前端使用jquery对上传文件做部分初步的判断,验证通过的文件利用ajaxFileUpload上传到服务器端,并将文件的存储路径保存到数据库。<asp:FileUploadI...
- Nodejs之MEAN栈开发(四)-- form验证及图片上传
-
这一节增加推荐图书的提交和删除功能,来学习node的form提交以及node的图片上传功能。开始之前需要源码同学可以先在git上fork:https://github.com/stoneniqiu/R...
- 大数据开发基础之JAVA jquery 大数据java实战
-
上一篇我们讲解了JAVAscript的基础知识、特点及基本语法以及组成及基本用途,本期就给大家带来了JAVAweb的第二个知识点jquery,大数据开发基础之JAVAjquery,这是本篇文章的主要...
- 推荐四个开源的jQuery可视化表单设计器
-
jquery开源在线表单拖拉设计器formBuilder(推荐)jQueryformBuilder是一个开源的WEB在线html表单设计器,开发人员可以通过拖拉实现一个可视化的表单。支持表单常用控件...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)