堆排序(向下调整法,向上调整法详解)

2024-03-18 16:44

本文主要是介绍堆排序(向下调整法,向上调整法详解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一、 二叉树的顺序结构

二、 堆的概念及结构

三、数组存储、顺序存储的规律

此处可能会有疑问,左右孩子的父节点计算为什么可以归纳为一个结论了?

四、大小堆解释

五、大小堆的实现(向上和向下调整法)

5.11向上调整法

 ​编辑

5.12向上调整法时间复杂度计算

5.21向下调整法

5.22向下调整法的时间复杂度计算

​编辑

六、堆排序的实现

代码如下:


一、 二叉树的顺序结构

普通的二叉树是不适合用数组来存储的,因为可能会存在大量的空间浪费。而完全二叉树更适合使用顺序结构存储。现实中我们通常把堆(一种二叉树)使用顺序结构的数组来存储,需要注意的是这里的堆和操作系统虚拟进程地址空间中的堆是两回事,一个是数据结构,一个是操作系统中管理内存的一块区域分段。

二、 堆的概念及结构

如果有一个关键码的集合k ={ k_0{}^{},k_1{}^{},k_2{}^{},...,k_n{}^{} },把它的所有元素按完全二叉树的顺序存储方式存储在一个一维数组中,并满足:

K_i <= K_{2*i+1}K_i <= K_{2*i+2}(K_i >= K_{2*i+2}K_i>=K_{2*i+2})i = 0,
2…,则称为小堆(或大堆)。将根节点最大的堆叫做最大堆或大根堆,根节点最小的堆叫做最小堆或小根堆。

  • 堆中某个节点的值总是不大于或不小于其父节点的值;
  • 堆总是一棵完全二叉树(完全二叉树是从满二叉树中最后一层连续删除若干结点(只能从最右侧删除)后得到的二叉树。)。

要满足K_i <= K_{2*i+1}K_i <= K_{2*i+2}(K_i >= K_{2*i+2}K_i>=K_{2*i+2})的原因。

三、数组存储、顺序存储的规律

如果要用数组存储二叉树,那么必须要符合顺序存储中父子存储的规律

此处可能会有疑问,左右孩子的父节点计算为什么可以归纳为一个结论了?

  • 一个左子节点索引 leftchild 和一个右子节点索引 rightchild,并且它们共享同一个父节点时,这意味着 rightchild = leftchild + 1。现在,如果你用上述公式来计算它们的父节点索引:
  • 对于左子节点:parent = (leftchild - 1) / 2
  • 对于右子节点:parent = (rightchild - 2) / 2但因为 rightchild = leftchild + 1,所以:
  • parent = ((leftchild + 1) - 2) / 2
  • parent = (leftchild - 1) / 2
  • 并且由于(int)3/(int)2 = (int)1,这一向下取整的性质,所以在这一计算过程中不会出现浮点数的情况
  • 你可以看到,无论你是从左子节点还是右子节点开始计算,你都得到了相同的父节点索引。

但是数组存储二叉树是有要求的。如果不符合该规律,那么得设置空节点去代替缺失的节点(因为要满足下标的规律才能方便查找),那么使用太多的空节点会造成空间的浪费。

结论:数组存储只适合完全二叉树和满二叉树

四、大小堆解释

 

堆并非是一定有序的 :左孩子与右孩子之间没有大小关系

  • 大堆:在最大堆中,父节点的值总是大于或等于其子节点的值。但是,左孩子和右孩子之间并没有固定的大小关系。也就是说,左孩子可以大于、小于或等于右孩子,这都不会违反最大堆的定义。
  • 也就是说,对于给定的节点i,其值应满足:array[i] >= array[2i + 1] 且 array[i] >= array[2i + 2]。
  • 小堆:在最小堆中,父节点的值总是小于或等于其子节点的值。同样地,左孩子和右孩子之间的大小关系是不确定的。
  • 也就是说,对于给定的节点i,其值应满足:array[i] <= array[2i + 1] 且 array[i] <= array[2i + 2]。
  • 这里的“2i + 1”和“2i + 2”分别表示节点i的左子节点和右子节点在数组中的位置(假设数组是从0开始索引的)。

这种特性使得堆成为一种非常有效的数据结构,特别是在实现优先队列等应用中。堆可以在对数时间内完成插入和删除最大(或最小)元素的操作,这是因为它不需要保持整个结构的完全排序。

举个例子:

    10  /   \  5     8  / \   / \  
2   3 6   7

在这个堆中,父节点的值总是大于或等于其子节点的值。但是,你可以看到左孩子和右孩子之间的大小关系是不一致的。例如,5的左孩子是2,右孩子是3,而8的左孩子是6,右孩子是7。这里并没有规定左孩子必须大于或小于右孩子。 

五、大小堆的实现(向上和向下调整法)

void Swap(HPDataType* px,HPDataType* py)
{*py ^= *px;*px ^= *py;*py ^= *px;
}

5.11向上调整法

目的:
当向堆中插入新元素时,为了维护堆的性质,需要对该元素进行向上调整。向上调整法就是从新插入的节点开始,通过与其父节点的比较和交换,确保该节点的值不大于(对于大根堆)或不小于(对于小根堆)其父节点的值。

步骤:

  1. 插入数据
  2. 与自己的父亲比较
  3. 交换/不交换
  4. 交换:孩子来到父亲位置,父亲来到自己父亲的位置。

判断条件:a[child] > a[parent]

结束循环条件:child > 0  (确保不是根节点)

时间复杂度:O(logN),其中N是堆中元素的数量。因为每次调整都涉及沿着树的一条路径向上移动,而树的深度为logN。

 

 void AdjustUP(HPDataType* a, int n, int parent)参数的意义:

  • HPDataType是一个自定义的数据类型,代表堆中存储的数据的类型int,a是一个指向HPDataType类型数组的指针,这个数组存储了堆中的所有元素。
  • child表示当前要进行向上调整的节点的索引。在堆排序中,当我们向堆中插入一个新的元素时,这个新元素通常被放置在数组的末尾,然后可能需要通过向上调整来确保它满足堆的性质。child就是这个新插入元素的索引。
void AdjustUp(HPDataType* a, int child)
{int parent = (child - 1) / 2;// 获取父节点索引//while (parent >= 0)while(child > 0)// 确保不是根节点{ //if (a[child] < a[parent])// 孩子小于于父亲,需要交换,向下调整法if (a[child] > a[parent])// 孩子大于父亲,需要交换, 向上调整法// 如果孩子节点大于父节点,则交换{Swap(&a[child], &a[parent]);child = parent;// 移动到父节点parent = (parent - 1) / 2;}else {break;}}}

5.12向上调整法时间复杂度计算

可得高度与向上调整的关系 F(h)=2^h*(h-2)+2

时间复杂度F(N)=(N+1)*(log(N+1)-2)+2

5.21向下调整法

目的:
当从堆中移除元素(通常是堆顶元素)后,为了维护堆的性质,需要对剩余的元素进行重新调整。向下调整法就是从父节点开始,通过与其子节点的比较和交换,确保父节点的值不大于(对于大根堆)或不小于(对于小根堆)其子节点的值。

步骤:

  1. 删除堆顶元素
  2. 堆顶元素与最后一个元素交换
  3. 删除最后一个元素
  4. 堆顶元素与左右两个孩子(最小/最大的孩子比较)
  5. 判断交换/不交换
  6. 交换:父亲来到孩子位置,孩子来到自己孩子的位置

判断条件:child + 1 < n && a[child + 1] < a[child]

结束循环条件:child < n(确保左孩子存在)

时间复杂度:O(logN),其中N是堆中元素的数量。因为每次调整都涉及沿着树的一条路径向下移动,而树的深度为logN。

如何删除堆顶数据后插入数据?

向下调整法

如果直接挪动覆盖:操作的时间复杂度太大,关系太乱,不如重新建堆

向下调整法:

 void AdjustDown(HPDataType* a, int n, int parent)参数的意义:

  • HPDataType是一个自定义的数据类型,代表堆中存储的数据的类型int,a是一个指向HPDataType类型数组的指针,这个数组存储了堆中的所有元素。
  • n表示堆中当前最后一个元素的下标。在堆排序的过程中,堆的大小可能会变化,因为我们会不断地从堆中移除元素。这个参数确保我们知道何时停止向下调整,即当child索引超过最后一个下标时。
  • parent表示当前要调整的节点的索引。在堆排序中,当我们从堆中移除堆顶元素并与堆的最后一个元素交换时,我们需要对新的堆顶元素进行向下调整以确保堆的性质得到维护。parent就是这个需要进行调整的节点的索引。
// 向下调整算法(用于删除或构建堆时维护堆)  
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; // 不需要交换,退出循环  }}
}

5.22向下调整法的时间复杂度计算

可得高度与向下调整次数的关系 F(h)=2^{h}-h-1

可得时间复杂度:F(N) = N-log(N+1)

六、堆排序的实现

有一个数列,请用堆排序升序排列

如果使用向下调整法建小堆,先把0视为堆根,0和3交换,然后当3视为堆根时:

所以要建大堆:

堆排序的时间复杂度与向上调整法建堆时差不多

子节点大于父节点时交换,建大堆,升序,保证父节点小于子节点

子节点小于父节点时交换,建小堆,降序,保证父节点大于子节点 

代码如下:

#include<bits/stdc++.h>
using namespace std;void Swap(int* px, int* py)
{*py ^= *px;*px ^= *py;*py ^= *px;
}
  • 该函数是堆排序的核心,用于调整堆的结构,确保其满足堆的性质(父节点小于其子节点,这是小根堆;反之则是大根堆。这里的代码是小根堆的实现)。
  • 接收三个参数:一个整数数组a、数组的长度n以及要调整的父节点的索引parent。
  • 首先,计算左孩子的索引child。
  • 然后,通过循环,比较父节点和孩子节点的大小。如果存在右孩子且右孩子的值小于左孩子,则选择右孩子作为更小的孩子。
  • 如果更小的孩子的值小于父节点,则交换它们的值,并将parent移动到新的位置,再次检查新的子节点。
  • 如果子节点不小于父节点,则循环终止,调整完成。
// 向下调整算法(用于删除或构建堆时维护堆)  
void AdjustDown(int* 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; // 不需要交换,退出循环  }}
}
  • 首先,对数组a建立一个小根堆。从最后一个非叶子节点开始(即索引为(n-1-1)/2的节点),调用AdjustDown函数调整每个子树。
  • 一旦堆建立完毕,进入循环:将堆顶元素(数组的第一个元素)与堆的最后一个元素交换,然后重新调整剩下的元素为堆,但每次调整的范围都减小一个(即排除掉最后一个元素)。
  • 循环继续,直到堆的大小为1,此时数组已经完全排序。
void HeapSort(int* a, int n)
{// a数组直接建堆 O(N)for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, n, i);}// O(N*logN)int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);// 首尾交换AdjustDown(a, end, 0);// 向下调整--end;}
}

这个函数首先通过AdjustDown函数将数组转化为最大堆。然后,它反复地将堆的根节点(即最大元素)与堆的最后一个节点交换,并重新调整堆,直到整个数组被排序。

int main()
{int a[] = { 3,6,1,5,8,9,2,7,4,0 };HeapSort(a, sizeof(a) / sizeof(int));for (int i = 0; i < 10; i++)printf("%d", a[i]);return 0;
}

今天就先到这了!!!

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!

你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。

这篇关于堆排序(向下调整法,向上调整法详解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Redis 的 SUBSCRIBE命令详解

《Redis的SUBSCRIBE命令详解》Redis的SUBSCRIBE命令用于订阅一个或多个频道,以便接收发送到这些频道的消息,本文给大家介绍Redis的SUBSCRIBE命令,感兴趣的朋友跟随... 目录基本语法工作原理示例消息格式相关命令python 示例Redis 的 SUBSCRIBE 命令用于订

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

MySQL8 密码强度评估与配置详解

《MySQL8密码强度评估与配置详解》MySQL8默认启用密码强度插件,实施MEDIUM策略(长度8、含数字/字母/特殊字符),支持动态调整与配置文件设置,推荐使用STRONG策略并定期更新密码以提... 目录一、mysql 8 密码强度评估机制1.核心插件:validate_password2.密码策略级

从入门到精通详解Python虚拟环境完全指南

《从入门到精通详解Python虚拟环境完全指南》Python虚拟环境是一个独立的Python运行环境,它允许你为不同的项目创建隔离的Python环境,下面小编就来和大家详细介绍一下吧... 目录什么是python虚拟环境一、使用venv创建和管理虚拟环境1.1 创建虚拟环境1.2 激活虚拟环境1.3 验证虚

详解python pycharm与cmd中制表符不一样

《详解pythonpycharm与cmd中制表符不一样》本文主要介绍了pythonpycharm与cmd中制表符不一样,这个问题通常是因为PyCharm和命令行(CMD)使用的制表符(tab)的宽... 这个问题通常是因为PyCharm和命令行(CMD)使用的制表符(tab)的宽度不同导致的。在PyChar

sky-take-out项目中Redis的使用示例详解

《sky-take-out项目中Redis的使用示例详解》SpringCache是Spring的缓存抽象层,通过注解简化缓存管理,支持Redis等提供者,适用于方法结果缓存、更新和删除操作,但无法实现... 目录Spring Cache主要特性核心注解1.@Cacheable2.@CachePut3.@Ca

SpringBoot请求参数传递与接收示例详解

《SpringBoot请求参数传递与接收示例详解》本文给大家介绍SpringBoot请求参数传递与接收示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋... 目录I. 基础参数传递i.查询参数(Query Parameters)ii.路径参数(Path Va