数据结构与算法:堆排序和TOP-K问题

朋友们大家好,本节内容来到堆的应用:堆排序和topk问题

堆排序

  • 1.堆排序的实现
    • 1.1排序
  • 2.TOP-K问题
  • 3.向上调整建堆与向下调整建堆
    • 3.1对比两种方法的时间复杂度

我们在c语言中已经见到过几种排序,冒泡排序,快速排序(qsort)

冒泡排序的时间复杂度为O(N2),空间复杂度为O(1);qsort排序的时间复杂度为
O(nlogn),空间复杂度为O(logn),而今天所讲到的堆排序在时间与空间复杂度上相比于前两种均有优势

堆排序可以在原数组上进行,其空间复杂度为O(1);
堆排序提供了稳定的 (O(nlogn)) 时间复杂度

接下来我们进行讲解

首先我们来看这组代码:

int main()
{int a[] = { 6,3,5,7,11,4,9,13,1,8,15 };Heap hp;HeapInit(&hp);for (int i = 0; i < sizeof(a) / sizeof(int); i++){HeapPush(&hp, a[i]);}while (!HeapEmpty(&hp)){printf("%d ", HeapTop(&hp));HeapPop(&hp);}printf("\n");return 0;}

上节课我们知道,hp这个堆里面,a[i]并不一定是有序的

在这里插入图片描述
这里我们每次打印首元素,即最小元素,再删除掉,下一次获得到的堆顶元素仍为最小的,所以打印出来结果为有序的。但这个并不是堆排序,他只是每次获取堆顶最小元素

堆排序是直接在数组上实现的

1.堆排序的实现

堆排序的实现可以分为两部分:构建最大堆(或最小堆)和执行排序过程

首先我们来看建堆过程

在上述代码中,我们是通过HeapPush(&hp, a[i]);来实现堆的插入,推其本质,是每次插入元素后进行向上调整,我们构建一个堆排序函数,其参数为传入的数组,和数组的元素个数:

void HeapSort(HPDataType* a, int n);

首先建堆,这里我们用向上调整建堆,在文章末尾会给大家引入向下调整建堆

for (int i = 1; i < n; i++)
{Ajustup(a, i);
}

从第二个元素开始,每次向上调整,完成堆的构建
在这里插入图片描述

建好之后我们则需要排序

1.1排序

思考一下,如果我们想要进行升序排序,需要建立大堆还是小堆呢?

在上述示例中,如果我们想进行升序,该怎么操作???

这里,如果我们想要升序排序,则需要建立大堆

小堆如果我们想要升序,堆顶元素在对应位置,剩余元素重新建立小堆,则时间复杂度大大增加

上述示例中,我们建了一个小堆,可以将Ajustup父节点与子节点大小关系改变来建立为大堆:

在这里插入图片描述
那思考一下,建立了大堆,我们如何实现升序呢?

这里我们就需要与删除堆顶元素相同的思路

  1. 排序过程
    在大堆构建完成后,数组的根节点(即数组的第一个元素)是当前堆中的最大元素。通过将它与堆的最后一个元素交换,然后减少堆的大小(实际上是忽略数组的末尾元素),可以确保最大元素位于数组的正确位置上。

  2. 调整堆
    交换根节点和最后一个节点之后,新的根节点可能破坏了大堆的性质,因此需要进行调整。调整的方法是将新的根节点“下沉”,直到恢复大堆的性质。

  3. 重复过程
    重复对堆顶元素进行移除并调整堆的过程,直到堆的大小减少到1。在每一次重复过程中,都会将当前的最大元素放置到它在数组中的最终位置上。

所以我们代码实现就两步:

  • 交换首尾元素
  • 向下调整
void HeapSort(HPDataType* a, int n)
{//建堆for (int i = 1; i < n; i++){Ajustup(a, i);}while (n>1){Swap(&a[0], &a[n - 1]);n--;Ajustdown(a, n, 0);}
}

我们进行代码测试
在这里插入图片描述

所以,堆这里可以促进我们快速选数,它的本质是选择排序

2.TOP-K问题

TOP-K问题指的是从一个大规模的数据集中找出“最重要”或“最优”的K个元素的问题,对于Top-K问题,能想到的最简单直接的方式就是排序,但是:如果数据量非常大,排序就不太可取了(可能数据都不能一下子全部加载到内存中)。最佳的方式就是用堆来解决

思路如下:

  1. 用数据集合中前K个元素来建堆
    • 前k个最大的元素,则建小堆
    • 前k个最小的元素,则建大堆
  2. 用剩余的N-K个元素依次与堆顶元素来比较,不满足则替换堆顶元素

基于已经提供的堆操作函数,我们可以实现一个特定的PrintTopK函数,其目的是从数组a中找到最大的k个元素。

实现这一目标的思路是:

  • 首先,使用数组a中的前k个元素建立一个最小堆。
  • 然后,遍历剩余的n-k个元素。对于每个元素,如果它大于堆顶元素,则用它替换堆顶元素,然后对堆顶元素进行向下调整以维护最小堆的性质。
  • 遍历完成后,堆中的k个元素即为整个数组中最大的k个元素。
void PrintTopK(int* a, int n, int k)
{Heap php;HeapInit(&php);for (int i = 0; i < k; ++i) {HeapPush(&php, a[i]);}for (int i = k; i < n; ++i) {if (a[i] > HeapTop(&php)) { // 如果当前元素比堆顶大HeapPop(&php); // 移除堆顶HeapPush(&php, a[i]); // 将当前元素加入堆中}}// 打印堆中的元素,即TOP K元素for (int i = 0; i < k; ++i) {printf("%d ", php.a[i]);}printf("\n");HeapDestroy(&php);
}
  1. 用a中前k个元素建立堆
  2. 将剩余n-k个元素与堆顶比较,替换并调整

测试代码:
在这里插入图片描述

3.向上调整建堆与向下调整建堆

对于数组a,进行向上调整建堆:

for (int i = 1; i < n; i++)
{Ajustup(a, i);
}

要通过向下调整的方式建立堆,我们通常是从最后一个非叶子节点开始,逐层向上进行调整,这能保证每个子树都满足堆的性质

for (int i = n/2 - 1; i >= 0; i--) {AdjustDown(a, n, i);}

3.1对比两种方法的时间复杂度

向下调整建堆

这个方法从最后一个非叶子节点开始,逆序对数组中的元素执行向下调整的操作。每个节点需要执行的向下调整操作取决于其高度,而数组中大约一半的节点是叶子节点,它们不需要被向下调整。对于剩下的节点,只有很少的节点需要移动到树的较低层次。具体地说,树的每一层上的节点数量减半,而向下移动的最大深度从0开始线性增加。

for (int i = n/2 - 1; i >= 0; i--) {AdjustDown(a, n, i);
}

在这里插入图片描述
设向下调整的累计次数为T(h).

  • 倒数第二层调整次数:2h-2*1
  • 倒数第三层调整次数:2h-3*2
  • ……
  • 第一层调整次数:20*(h-1);

对其进行累加和:

为等差×等比求和,通过错位相减则可求出结果:

T(h)=2^h-1-h;
h=log (n+1);
T(n)=n-log(n+1)

导致最大影响的项为n
所以向下调整的时间复杂度为O(N)

向上调整建堆

从第二层开始向上调整:

  • 第二层调整次数:21*1
  • 第三层调整次数:22*2;
  • 倒数第二层:2h-2*(h-2);
  • 倒数第一层:2h-1*(h-1);

向上调整建堆

对于一个节点来说,向上调整可能需要比较和移动直到它的根节点,这在最坏的情况下是树的高度,对于一个完全二叉树来说,树的高度是 O ( log ⁡ n ) O(\log n) O(logn)。对于代码段:

for (int i = 1; i < n; i++) {AdjustUp(a, i);
}

这个方法从第二个元素开始,逐一对数组中的元素执行向上调整的操作。对于数组中的第i个元素,最坏情况下向上调整操作需要沿着一条从叶节点到根节点的路径移动,路径的长度大约等于树的高度 h h h,即 O ( log ⁡ i ) O(\log i) O(logi)。因此,对于所有元素的总时间复杂度为:

T ( n ) = ∑ i = 1 n O ( log ⁡ i ) = O ( log ⁡ n ! ) = O ( n log ⁡ n ) T(n) = \sum_{i=1}^{n} O(\log i) = O(\log n!) = O(n \log n) T(n)=i=1nO(logi)=O(logn!)=O(nlogn)

使用斯特灵公式( n ! ≈ 2 π n ( n e ) n n! \approx \sqrt{2\pi n}(\frac{n}{e})^n n!2πn (en)n),可以推导出 O ( log ⁡ n ! ) O(\log n!) O(logn!) 的大致等于 O ( n log ⁡ n ) O(n \log n) O(nlogn),所以向上调整建堆的时间复杂度大约为 O ( n log ⁡ n ) O(n \log n) O(nlogn)

向上调整建堆的时间复杂度是 O ( n log ⁡ n ) O(n \log n) O(nlogn),而向下调整建堆的时间复杂度是 O ( n ) O(n) O(n)。因此,对于从零开始构建堆的场景,通常更倾向于使用向下调整的方法,因为它更加高效。

本节内容到此结束!感谢大家支持!

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

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

相关文章

Nodejs 第四十九章(lua)

lua Lua是一种轻量级、高效、可嵌入的脚本语言&#xff0c;最初由巴西里约热内卢天主教大学&#xff08;Pontifical Catholic University of Rio de Janeiro&#xff09;的一个小团队开发而成。它的名字"Lua"在葡萄牙语中意为"月亮"&#xff0c;寓意着Lua…

浏览器中的HTTP请求原理

一.浏览器的多进程架构(Chrome) Chrome打开一个页面需要启动多少进程&#xff1f;我们可以点击Chrome浏览器右上角的“选项”菜单&#xff0c;选择“更多工具”子菜单&#xff0c;点击“任务管理器”&#xff0c;这将打开Chrome的任务管理器的窗口&#xff1a; Chrome任务管…

Java_优先级队列(堆)(Priority Queue)

文章目录 一、优先级队列1.概念 二、优先级队列的模拟1.堆的概念2.堆的存储方式3.堆的创建1、堆向下调整2、堆的创建代码实现3、建堆的时间复杂度 2.堆的插入与删除1、堆的插入2、堆的删除3、完整的堆代码4、练习 一、PriorityQueue常用接口介绍1.PriorityQueue的特性2.Priorit…

如何使用程序调用通义千问

之前分享了&#xff0c;使用程序调用文心一言。但是很快文心一言就要收费了。阿里的提供了暂时免费版的基础模型&#xff0c;效果还算可以。所以再分享一下&#xff0c;如何使用程序来调用通义千问的模型。 整体很简单&#xff0c;分三步&#xff1a;导入依赖&#xff1b;获取A…

基于springboot+vue的酒店管理系统

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

MySQL的三大范式

文章目录 简介第一范式第二范式第三范式&#xff1a; 简介 在MySQL的使用中&#xff0c; 要根据实际灵活设计表&#xff0c;一般来说我们通常遵循三大范式&#xff08;啥是范式&#xff1a;是一些约束、规范、规则&#xff0c; 来优化数据库表的设计和存储&#xff09;,三大范…

激活函数Swish(ICLR 2018)

paper&#xff1a;Searching for Activation Functions 背景 深度网络中激活函数的选择对训练和任务表现有显著的影响。目前&#xff0c;最成功和最广泛使用的激活函数是校正线性单元&#xff08;ReLU&#xff09;。虽然各种手工设计的ReLU替代方案被提出&#xff0c;但由于在…

更改Linux系统(我的是centOS7)[root@localhost ~]$的字体颜色样式

打开根目录 cd ~ 编辑样式文件 vi .bashrc 出现了一串代码 在末尾加入 PS1"\[\e[37;1m\][\[\e[35;1m\]\u\[\e[37;1m\]\[\e[32;1m\]\h \[\e[34;1m\]\W\[\e[37;1m\]]\[\e[33;1m\]\$ \[\e[0m\]" 保存并退出 最后刷新一下文件 source .bashrc 然后就发现 非常完…

构建信息蓝图:概念模型与E-R图的技术解析

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua&#xff0c;在这里我会分享我的知识和经验。&#x…

【JavaEE】_Spring MVC项目之使用对象传参

目录 1. 使用对象传参 2. 后端参数重命名问题 2.1 关于RequestPara注解 本专栏关于Spring MVC项目的单个及多个参数传参一文中&#xff0c;已经介绍过了对于不同个数的参数传参问题&#xff0c;原文链接如下&#xff1a; 【JavaEE】_Spring MVC 项目单个及多个参数传参-CSD…

安装RabbitMQ及配置Centos7 方式(2)

1、背景需求 自行搭建学习参考使用&#xff0c;这里采用的Centos7 方式&#xff0c;这已经是多年前的方式了&#xff0c;现在主流方式是容器化安装、部署&#xff0c;docker、ks8&#xff0c;同学们可自行去学习参考。 2、搭建环境 环境&#xff1a;centos7 、otp_src_21.3、…

Java ElasticSearch面试题

Java ES-ElasticSearch面试题 前言1、ElasticSearch是什么&#xff1f;2. 说说你们公司ES的集群架构&#xff0c;索引数据大小&#xff0c;分片有多少 &#xff1f;3. ES的倒排索引是什么&#xff1f;4. ES是如何实现 master 选举的?5. 描述一下 ES索引文档的过程&#xff1a;…