【C】堆的应用1 -- 堆排序

news/2025/2/25 6:54:54

  之前学习了堆,堆的一棵以顺序结构存储的完全二叉树,堆本身又氛围大根堆和小根堆,假设以大根堆为例,由于堆顶部元素是一棵二叉树里面最大的元素,所以如果每次都取堆顶的元素,那么取出的元素就是一个降序排列的序列。至此,我们发现了一个堆的特别重要的一个应用,就是堆排序。


目录

1  问题解析

算法分析

3  代码 

4  时空复杂度分析

(1) 时间复杂度

(2) 空间复杂度


1  问题解析

  堆排序顾名思义就是用堆结构来实现对一个数组的排序,但是在排序过程中是不能使用堆这个数据结构的。如:有一个数组 a[] = {2, 4, 10, 9,  2,  3},通过堆排序这个排序算法之后,数组 a 里面的数据变为了 a[] = {2, 2, 3, 4, 9, 10} 升序排列;或者是 a[]  = {10, 9, 4, 3, 2, 2} 降序排列。


算法分析

  该算法共分为两步:

  1) 首先对于给定的一个数组,数组里面的数据都是乱序的,既然是堆排序,我们就需要先让该数组里面的数据变成一个堆,将数组中的数据变成一个堆的算法为(这里是建大堆,建小队逻辑类似):
  这里利用的是向下调整算法,因为向下调整算法的时间效率是比向上调整算法的时间效率高的(上一篇文章有所讲解),而向下调整算法又需要其左右子树各自都是一个堆,所以首先选择最后一个节点作为孩子节点,让其父亲向下调整,这样最后一棵子树就变成了一个堆,然后再以当前孩子节点的上一个节点作为下一个孩子节点,让其父亲向下调整,直至所有的子树都被调整为了一个堆。该过程如图所示:

 

其本质就是从最小的一个子树建堆,然后使子树规模依次扩大,最后扩大为整棵树。

  2)数组建完堆之后,由于堆顶数据总是最大的,所以我们选择让堆顶数据和最后一个节点的数据进行交换,这样最后一个数据就是有序的,然后让交换后的堆顶数据再向下建堆,但是注意这次建堆的范围应是 n - 1 个数据(n 为数组中数据的个数);那么当前我们执行完 n - 1 次该操作之后,最大的 n - 1 个数据就会被升序的放到后 n - 1 个位置,所以就实现了升序排列。该算法如图所示:

  通过理解该算法,如果要排升序的话,那么应该建大堆;排降序的话,应该建小堆。


3  代码 

//升序的话,建大堆;降序的话,建小堆
//这里升序排列
void Swap(int* x, int* y)
{
  int tmp = *x;
  *x = *y;
  *y = tmp;
}

void AdjustDown(int* arr, int parent, int n)
{
  int child = 2 * parent + 1;
  while (child < n)
  {
    if (arr[child + 1] > arr[child] && child + 1 < n)
    {
      child++;
    }
    if (arr[child] > arr[parent])
    {
      Swap(&arr[child], &arr[parent]);
      parent = child;
      child = 2 * parent + 1;
    }
    else
    {
      break;
    }
  }
}

//Heap是堆的意思,Sort是排序的意思
void HeapSort(int* arr, int n)
{
  //先用所给的数组向下调整建一个堆
  for (int i = (n-1-1)/2;i >= 0;i--)
  {
    AdjustDown(arr, i, n);
  }
  //然后每次交换堆顶元素和最后一个元素,让剩余元素建大堆
  int end = n;
  while (end > 0)
  {
    //先交换
    Swap(&arr[0], &arr[--end]);
    //然后剩下元素建大堆
    AdjustDown(arr, 0, end);
  }
}

  

测试用例:

void Test()
{
    int arr[] = { 10, 9, 2, 4, 6, 11, 56, 20, 15, 19};
    int n = sizeof(arr) / sizeof(arr[0]);
    HeapSort(arr, n);
    for (int i = 0; i < n; i++)
    {
        printf("%d ", arr[i]);
    }
}

int main()
{
    Test();
    return 0;
}

4  时空复杂度分析

(1) 时间复杂度

  在上一篇文章中,我们分析了向下调整算法的时间复杂度为 O(logn) 的,而在堆排序里面共有两次循环,第一次循环会执行 n/2 - 1 次,第二次循环会执行 n 次,而这两次循环里面都嵌套了向下调整算法,所以时间复杂度为 O( (n/2 - 1)*logn + nlogn),去除常数项和低阶项,堆排序时间复杂度就是 O(nlogn)的。

(2) 空间复杂度

  由于堆排序算法仅用了几个变量,并为额外开辟空间,所以空间复杂度为 O(1)。


http://www.niftyadmin.cn/n/5865121.html

相关文章

量子计算的数学基础:复数、矩阵和线性代数

量子计算是基于量子力学原理的一种新型计算模式,它与经典计算机在信息处理的方式上有着根本性的区别。在量子计算中,信息的最小单位是量子比特(qubit),而不是传统计算中的比特。量子比特的状态是通过量子力学中的数学工具来描述的,因此,理解量子计算的数学基础对于深入学…

【Java项目】基于Spring Boot的家具销售电商系统

【Java项目】基于Spring Boot的家具销售电商系统 技术简介&#xff1a;采用Spring Boot框架、Java技术、MySQL数据库等实现。 系统简介&#xff1a;家具销售电商系统主要实现了管理员模块、用户模块二大部分。1、管理员&#xff1a;首页、个人中心、家具分类管理、热销家具管理…

DeepSeek技术全景解析:架构创新与行业差异化竞争力

一、DeepSeek技术体系的核心突破 架构设计&#xff1a;效率与性能的双重革新 Multi-head Latent Attention (MLA)&#xff1a;通过将注意力头维度与隐藏层解耦&#xff0c;实现显存占用降低30%的同时支持4096超长上下文窗口。深度优化的MoE架构&#xff1a;结合256个路由专家…

ONNX转RKNN的环境搭建和部署流程

将ONNX模型转换为RKNN模型的过程记录 工具准备 rknn-toolkit:https://github.com/rockchip-linux/rknn-toolkit rknn-toolkit2:https://github.com/airockchip/rknn-toolkit2 rknn_model_zoo:https://github.com/airockchip/rknn_model_zoo ultralytics_yolov8:https://github…

解锁CSnakes:.NET与Python的融合魔法

一、引言 在软件开发的广袤领域中&#xff0c;我们常常面临各种复杂的业务需求和技术挑战。不同的编程语言犹如各具特色的工具&#xff0c;它们在不同的场景下展现出独特的优势。例如&#xff0c;C# 以其强大的类型系统和丰富的类库&#xff0c;在企业级应用开发中占据重要地位…

【JavaWeb学习Day19】

Tlias智能学习系统&#xff08;员工管理&#xff09; 删除员工&#xff1a; 需求分析&#xff1a; 其实&#xff0c;删除单条数据也是一种特殊的批量删除&#xff0c;所以&#xff0c;删除员工的功能&#xff0c;我们只需要开发一个接口就行了。 三层架构&#xff1a; Cont…

Oracle Fusion Middleware更改weblogic密码

前言 当用户忘记weblogic密码时&#xff0c;且无法登录到web界面中&#xff0c;需要使用服务器命令更改密码 更改方式 1、备份 首先进入 weblogic 安装目录&#xff0c;备份三个文件&#xff1a;boot.properties&#xff0c;DefaultAuthenticatorInit.ldift&#xff0c;Def…

【Microsoft PowerPoint for Mac】2分钟配置-MAC一键删除PPT中的所有备注

MAC一键删除PPT中的所有备注 1.搜索自动操作2.点击快速操作3.搜索并运行AppleScript4.输入代码&#xff0c;并选择只应用于Microsoft PowerPoint for Mac【右上角】5. CRTLS保存为“清除当前文稿中的所有备注”&#xff0c;PPT中应用。 MAC没自带&#xff0c;需要自己配置 1.搜…