【数据结构】堆排序详解

文章目录

  • 一、堆排序思想
  • 二、向上调整建堆排序
  • 三、向下调整建堆排序
  • 四、总结

对于什么是堆,堆的概念分类以及堆的向上和向下两种调整算法可见: 堆的创建

在这里插入图片描述

一、堆排序思想

int a[] = { 2,3,5,7,4,6 };

对于这样一个数组来说,要想要用堆排序对它进行排序,首先要做的就是用数组里的数据建立一个堆,大堆和小堆都可以,只有是一个堆才能使用堆排序。

那么应该建大堆还是小堆呢,例如对于这个数组要排升序,如果建立小堆的话
在这里插入图片描述

建成小堆后找出了最小的元素,要找到次小的就需要把剩下的元素看作堆,但剩下的元素不一定是堆,需要重新建堆,代价比较大。

更好的方法是升序建立大堆,堆顶和最后一个元素交换,最后一个最大的元素就已经有序,对剩下数据进行向下调整,就能找出第二大的,以此就能将数组排好序。

总结一下堆排序的思想就是:
1、根据要排什么序建大堆或小堆,此时堆顶端的元素就是最值
2、将顶端元素和末尾元素交换,此时末尾元素就是有序的,剩下的还有n-1个元素
3、将剩下的n-1个元素再次构建成堆,然后将堆顶端元素与第n-1个元素互换,反复执行便可得到有序数组

升序:建大堆
降序:建小堆

二、向上调整建堆排序

使用向上调整算法建堆的堆排序

例如:将数组a用堆排序按从小到大排列(升序)

在这里插入图片描述
首先,利用向上调整算法建大堆,此方法可参考堆的创建

向上调整算法的前提条件是:前面的元素是堆

对于单个结点来说既可以看作一个大堆,所以便可以通过向上调整算法依次对数组元素进行调整,那进行调整的元素前就一定是堆,满足条件

在这里插入图片描述

创建好的大堆如下:
在这里插入图片描述

将堆的顶端元素7和末尾元素2进行交换,对除7外剩下的元素进行向下调整重新构建大堆
在这里插入图片描述
此时7已经是有序的,将元素6和元素3进行交换,对除6、7外剩下元素进行向下调整重新构建大堆
在这里插入图片描述
此时6、7已经有序,将元素5和元素2进行交换,对除5、6、7外剩下元素进行向下调整重新构建大堆
在这里插入图片描述
此时5、6、7已经有序,将元素4和元素2进行交换,此时数组已经有序在这里插入图片描述
排序完数组a变为
在这里插入图片描述

向上调整算法建堆升序的堆排序代码如下:

#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
typedef int HPDataType;
//交换结点的函数
void Swap(HPDataType* p1, HPDataType* p2)
{HPDataType tmp = *p1;*p1 = *p2;*p2 = tmp;
}
//向上调整算法(大堆)
void AdjustUp(HPDataType* a, int child)
{//找出双亲的下标int parent = (child - 1) / 2;while (child>0){//孩子结点比双亲大则交换if (a[child] > a[parent]){Swap(&a[child], &a[parent]);child = parent;parent = (child - 1) / 2;}else{break;}}
}//向下调整算法(大堆)
void AdjustDown(HPDataType* a, int n, int parent)
{//先默认左孩子是较大的int child = parent * 2 + 1;while (child < n){//找出左右孩子中较大的if (child + 1 < n && a[child + 1] > a[child]){child++;}//孩子节点更小则交换if (a[child] > a[parent]){Swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else{break;}}
}
//排序
void HeapSort(int* a, int n)
{//向上调整建堆for (int i = 1; i < n; i++){AdjustUp(a, i);}//最尾端数据下标为总数减一int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);//对剩余元素进行向下调整AdjustDown(a, end, 0);end--;}
}
int main()
{int a[] = { 2,3,5,7,4,6 };HeapSort(a, sizeof(a) / sizeof(int));for (int i = 0; i < sizeof(a) / sizeof(int); i++){printf("%d ", a[i]);}return 0;
}

运行结果如下:
在这里插入图片描述

空间复杂度:O(1)
平均时间复杂度:O(nlogn)

三、向下调整建堆排序

向下调整建堆排序与向上调整建堆排序不同的地方就在于建堆时用的算法不同,建好堆之后的后续操作都是相同的。

还是对上面那个案例,我们用向下调整算法建堆
在这里插入图片描述

向下调整算法前提条件:左右子树必须是堆,才能调整

在这里插入图片描述

对于这个完全二叉树来说,它的倒数第一个非叶子节点2的左子树为4,没有右子树,可以用向下调整,再上一个节点6的左右子树是单个节点也可以看作堆,所有我们就可以从倒数第一个非叶子节点也就是最后一个节点的父亲开始向下调整:

在这里插入图片描述

利用向下调整建好堆之后的后续操作与向上调整建好堆之后的操作一样,这里就不再演示

向下调整算法建堆升序的堆排序代码更改如下:

void HeapSort(int* a, int n)
{向上调整建堆//for (int i = 1; i < n; i++)//{//	AdjustUp(a, i);//}// //向下调整建堆for (int i = (n - 1 - 1) / 2; i >= 0; i--){AdjustDown(a, n, i);}//最尾端数据下标为总数减一int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);//对剩余元素进行向下调整AdjustDown(a, end, 0);end--;}
}

利用向下调整建堆的堆排序时间复杂度为:O(n),比利用向上调整建堆更优

四、总结

使用堆排序需要先建堆,建堆有向上调整算法和向下调整算法两种方法,但向下调整算法的平均时间复杂度更低,建好堆之后便首尾数据互换,再对剩下元素重新建堆,反复执行便可得到有序数列。

重点知识总结:

  • 小堆:所有的双亲结点都小于孩子节点,根节点最小
  • 大堆:所有的双亲结点都大于孩子节点,根节点最大
  • 向下调整算法前提:左右子树必须是堆,才能调整
  • 向上调整算法前提:前面的元素是堆
  • 堆排序建堆时:升序建大堆,降序建小堆

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/110002.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

脚本:用python实现五子棋

文章目录 1. 语言2. 效果3. 脚本4. 解读5. FutureReference 1. 语言 Python 无环境配置、无库安装。 2. 效果 以第一回合为例 玩家X 玩家0 3. 脚本 class GomokuGame:def __init__(self, board_size15):self.board_size board_sizeself.board [[ for _ in range(board_…

(高阶)Redis 7 第12讲 数据双写一致性 经验篇

面试题 /问题1涉及到redis缓存和数据库双存储双写,出现的数据一致性问题,如何解决2双写一致性,先动缓存Redis还是数据库?3延时双删做过吗,有哪些问题?4微服务查询Redis无数据,数据库有数据,为保证数据双写一致性回写Redis需要注意什么?双检加锁策略了解吗?如何避免缓存…

操作系统运行环境

目录 操作系统运行环境&#xff08;1&#xff09; 第一节 一、处理器的构成与基本工作方式 二、特权和非特权指令 三、处理器的工作状态 四、程序状态字 第二节、计算机系统硬件部件 一、存储系统 二、I/O部件&#xff1a;1.I/O结构&#xff0c;2.通道&#xff0c;3.D…

CSS 四中方法实现水平 垂直居中

CSS 四中方法实现水平 垂直居中 效果 结构 <div class"a"><div class"b"></div></div>样式 方法一 .a {width: 500px;height: 500px;margin: 50px auto;background-color: crimson;// 核心代码display: flex;justify-content: …

HTML整站规划与规范

文章目录 命名规则命名命名书写 包含样式规范样式重置样式引入页面结构页面宽度页面高度与背景页面设计 网址图标 命名规则 命名 根据每块元素的主题、功能、页面上的位置命名&#xff0c;便于后期更改与维护。 另外&#xff1a;如果所有样式放在同一文件下&#xff0c;可以给…

idea项目配置三大步

场景&#xff1a; 使用 idea 打开一个新项目的时候&#xff0c;想让项目迅速跑起来&#xff0c; 其实只需要下面简单三步&#xff1a; 1. 首先&#xff0c;配maven 2. 其次&#xff0c;配置 jdk 这里配置 project 就行了&#xff0c;不用管Modules中的配置。 3. 最后&#…

ubuntu 22.04运行opencv4的c++程序遇到的问题

摘要&#xff1a;本文介绍一下在ubuntu系统中&#xff0c;运行一个最简单的opencv4程序都出问题的解决方法&#xff0c;并对其基本原理作简单阐述。解决问题的方法有很多&#xff0c;本文只提供其中一种。 opencv版本是4.2.0&#xff0c;ubuntu版本是20.04 查询opencv版本的指…

如何让两台手机相互远程控制?

你的两台手机是什么系统的&#xff1f;如果你的两台手机都是安卓系统&#xff0c;而且都是安卓7.0及以上版本的系统&#xff0c;那么恭喜你&#xff0c;这两台手机可以相互远程控制&#xff01; 你可以利用两个软件实现将两台手机相互远程控制的想法。为了避免混淆&#xff0c…

XL-LightHouse 与 Flink 和 ClickHouse 流式大数据统计系统

一个Flink任务只能并行处理一个或少数几个数据流&#xff0c;而XL-LightHouse一个任务可以并行处理数万个、几十万个数据流&#xff1b; 一个Flink任务只能实现一个或少数几个数据指标&#xff0c;而XL-LightHouse单个任务就能支撑大批量、数以万计的数据指标。 1、XL-LightHo…

c语言每日一练(15)

前言&#xff1a;每日一练系列&#xff0c;每一期都包含5道选择题&#xff0c;2道编程题&#xff0c;博主会尽可能详细地进行讲解&#xff0c;令初学者也能听的清晰。每日一练系列会持续更新&#xff0c;上学期间将看学业情况更新。 五道选择题&#xff1a; 1、程序运行的结果…

Linux四种I/O模型

一.四种模型 阻塞式IO&#xff0c;非阻塞式IO&#xff0c;信号驱动IO&#xff0c;IO多路复用 二.阻塞式IO 特点&#xff1a;最简单&#xff0c;最常用&#xff0c;效率低 阻塞I/O 模式是最普遍使用的I/O 模式 系统默认状态&#xff0c;套接字建立后所处于的模式就是阻塞I/O 模式…

IDEA连接数据库-MySql为例

1.找到Database 说明&#xff1a;找到Data Source下面的MySql 2.点击mysql 3.填写相关配置 说明&#xff1a;填写连接哪台服务器(这里是本机)&#xff0c;账号&#xff0c;密码&#xff0c;然后点击Test Connection 4.显示 说明&#xff1a;表示一共有7个&#xff0c;但是展…