数据结构和算法|堆排序系列问题(一)|堆、建堆和Top-K问题

news2025/1/12 20:46:30

在这里不再描述大顶堆和小顶堆的含义,只剖析原理层面。

主要内容来自:Hello算法

文章目录

  • 1.堆的实现
    • 1.1 堆的存储与表示过程
    • 1.2 访问堆顶元素
    • 1.4元素出堆
  • 2.⭐️建堆
    • 2.1 方法一:借助入堆操作实现
    • 2.2 ⭐️方法二:通过遍历堆化实现
  • 应用:Top-K问题
    • ⭐️方法一:遍历选择
    • 方法二:排序
    • ⭐️方法三:堆

1.堆的实现

1.1 堆的存储与表示过程

完全二叉树非常适合用数组来进行表示,而堆就是一颗完全二叉树,所以我们可以使用数组来存储堆。也就是说,堆的逻辑结构是一颗完全二叉树,它的物理结构(底层存储)是一个数组。

对于一个给定索引 i i i,其左子树和右子树索引分别为 2 i + 1 2i + 1 2i+1 2 i + 2 2i + 2 2i+2,其父节点的索引为 ( i − 1 ) / 2 (i-1)/2 (i1)/2(向下取整)。

封装索引的映射公式:

//获取左子节点的索引
int left(int i) {
	return 2 * i + 1;
}
//获取右子节点的索引
int right(int i) {
	return 2 * i + 2;
}
//获取父节点的索引
int parent(int i) {
	return (i - 1) / 2;
} 

1.2 访问堆顶元素

堆顶元素即为二叉树的根结点,也就是列表的首元素 ```cpp //访问堆顶元素 int peek() { return maxHeap[0]; } ``` ## ⭐️1.3元素入堆 该环节非常重要。

对于一个给定元素 val ,

  • 将其添加到堆底。添加到堆底后, val 可能大于堆中其他元素,所以我们需要恢复从出啊如结点到根结点的路径上的各个节点,这个操作就是堆化(heapify)
  • 从入堆结点开始,从底到顶执行堆化。我们比较插入节点与其父节点的值,如果插入节点更大,则将它们交换。然后继续执行此操作,从底至顶修复堆中的各个节点,直至越过根节点遇到无须交换的节点时结束。

动态流程可以查看文章:3. 元素入堆

我们应该分析一下入堆的时间复杂度,设节点总数为 n n n,则树的高度为 O ( l o g n ) O(logn) O(logn)。所以堆化操作的循环论述最多为 O ( l o g n ) O(logn) O(logn)
综上,元素入堆操作的时间复杂度为 O ( l o g n ) O(logn) O(logn)

void push(int val) {
	//添加节点minHeap是一个数组
	maxHeap.push_back(val);
	//从底至顶堆化
	siftUp(size() - 1);
}
//从节点i开始,从底至顶堆化操作
void siftUp(int i) {
	while (true) {
		//获取节点 i 的父节点
		int p = parent(i);
		//当“越过根节点”或“节点无须修复”时,结束堆化
		if (p < 0 || maxHeap[i] <= maxHeap[p]) 
			break;
		//交换两个结点
		swap(maxHeap[i], maxHeap[p]);
		//循环向上堆化
		i = p;
	}
}

再次强调:元素入堆的时间复杂度为 O ( l o g n ) O(logn) O(logn)

1.4元素出堆

元素出堆和我们一般想的所谓直接直接弹出不同,因为如果直接弹出,那么我想想要修复堆结构变得极其困难。为了尽量减少元素索引的变动,我们这样操作出堆:

  1. 交换堆顶元素与堆底元素(交换根结点与最右叶子结点)。
  2. 交换完成后,将堆底从列表中删除(由于已经交换,因此实际上删除的是原来的堆顶元素)。
  3. 从根结点开始,从顶至底执行堆化操作(下溯)。

“从顶至底堆化”的操作方向与“从底至顶堆化”相反,我们将根节点的值与其两个子节点的值进行比较,将最大的子节点与根节点交换。然后循环执行此操作,直到越过叶节点遇到无须交换的节点时结束。

具体流程可以看4. 堆顶元素出堆,流程极为详细。

时间复杂度分析,从顶至底的堆化,很明显时间复杂度仍然是O(logn)。

/* 元素出堆 */
void pop() {

/* 从节点 i 开始,从顶至底堆化 */
void siftDown(int i) {

2.⭐️建堆

给定你任意一个列表,我们想要使用其所有元素来构建一个堆,这个过程被称为“建堆操作”。

2.1 方法一:借助入堆操作实现

首先我们维护一个空堆,然后依次对每个元素执行“入堆操作”,即现将元素添加至堆的尾部,然后“从底至顶”堆化即可。

至此我们可以维护一个真正的堆了,而且肯定是根结点最先被构建出来。

每当一个元素入堆,堆的长度就加一。由于节点是从顶到底依次被添加进二叉树的,因此堆是“自上而下”构建的。

设元素数量为 n n n,每个元素的入堆操作使用 O ( l o g n ) O(logn) O(logn)时间,所以整个建堆方法的时间复杂度为:
O ( n l o g n ) O(nlogn) O(nlogn)

2.2 ⭐️方法二:通过遍历堆化实现

这是一个更高效的建堆方法,共分为两步:

  • 将列表所有元素原封不动添加到堆中,此时堆的性质尚未得到满足;
  • 倒序遍历堆(层序遍历的倒序),依次对每个非叶节点执行“从顶至底堆化”

每当堆化一个结点后,以该节点为根结点的子树就形成了一个合法的子堆。

而由于是倒序遍历,因此堆是“自下而上”构建的。(之所以选择倒序遍历,是因为这样能够保证当前节点之下的子树已经是合法的子堆,这样堆化当前节点才是有效的。)

并且由于叶子结点没有子结点,因此他们天然就是合法的子堆,无须堆化。

/* 构造方法,根据输入列表建堆 */
MaxHeap(vector<int> nums) {
	//将列表元素原封不动添加进堆
	maxHeap = nums;
	//堆化出也节点以外的其他所有节点
	for (int i = parent(size() - 1); i >= 0; i--) {
		siftDown(i);
	}
}

时间复杂度分析:

  • 假设完全二叉树的节点数量为 n n n,则叶节点数量为 ( n + 1 ) / 2 (n+1)/2 (n+1)/2。因此需要堆化的节点数量为 ( n − 1 ) / 2 (n-1)/2 (n1)/2
  • 从顶至底堆化的过程中,每个节点最多堆化到叶节点,因此最大迭代次数为二叉树高度 l o g n logn logn

将上述两者相乘,可得到建堆过程的时间复杂度为 O ( n l o g n ) O(nlogn) O(nlogn)

但这只是一种粗略的估算,严格来说,我们应该进行详细的计算,基本的思想就是高度较高的节点堆化需要的迭代次数较多,较低节点堆化的迭代次数较少。
我们需要对各层的“节点数量X结点高度”求和,得到所有节点的堆化迭代次数的总和
最终的结果是 O ( n ) O(n) O(n)

应用:Top-K问题

Question:
给定一个长度为n的无序数组nums,请返回数组中最大的k个元素。

⭐️方法一:遍历选择

暴力求解!我们进行 k 轮遍历,分别在每轮中提取第 1 、 2 、 . . . 、 k 1、2、...、k 12...k大的元素,时间复杂度为 O ( n k ) O(nk) O(nk)
当然了,此方法只适用于 k < < n k << n k<<n 的情况,因为当 k k k n n n比较接近时,其时间复杂度趋向于 O ( n 2 ) O(n^2) O(n2),非常耗时:

k = n k=n k=n 时,我们可以得到完整的有序序列,此时等价于“选择排序”算法。

算法步骤如下:

  • 初始化:指定一个变量来记录当前需要考虑的数组的结尾位置。
  • 重复寻找最大值:遍历当前未处理的数组部分,找到最大元素。
  • 交换元素: 将找到的最大元素与当前考虑的数组的最后一个元素交换。
  • 调整考虑范围: 减少考虑数组范围的长度。
  • 重复: 重复以上步骤 𝑘。
vector<int> topKsearch(vector<int>& nums, int k) {
	int n = nums.size();
	int end = n - 1 //初始化结束索引
	for (int i = 0; i < k; i++) {
		int maxIndex = 0;
		for (int j = 0; j <= end; j++) {
			if (nums[j] > nums[maxIndex]) 
				maxIndex = j;
		} 
		//交换最大元素到当前考虑的数组的末尾
		swap(nums[maxIndex], nums[end]);
		end--;
	}
}

方法二:排序

这个思路也比较简单,先对数组进行排序,然后返回最右边的k个 元素,时间复杂度为: O ( n l o g n ) O(nlogn) O(nlogn)
显然,该方法“超额”完成任务了,我们其实只需要找出最大的k个元素即可。

vector<int> topKsort(vector<int>& nums, int k) {
    sort(nums.begin(), nums.end(), greater<int>()); // 降序排序
    vector<int> result;
    for (int i = 0; i < k; i++) {
        result.push_back(nums[i]);
    }
    return result;
}

⭐️方法三:堆

堆天生就适合解决这样的Top-K问题。

  1. 初始化一个小顶堆,其顶堆元素最小;
  2. 现将数组的前 k k k个元素依次入堆,也就是说我们只维护大小为 k k k的堆;
  3. 从第 k + 1 k+1 k+1个元素开始,如果当前元素大于堆顶元素,则将堆顶元素出堆,并将当前元素入堆。
  4. 遍历整个nums后,堆中保存的就是最大的k个元素。

具体的实验流程可以看《Hello World》文章:方法三:堆

时间复杂度分析:
我们一共执行n次入堆和出堆,堆的最大长度为 k k k,所以时间复杂度为 O ( n l o g k ) O(nlogk) O(nlogk)
该方法效率极高,当 k k k较小时,时间复杂度趋向于 O ( n l o g k ) O(nlogk) O(nlogk);当 k k k较大时,时间复杂度也不会超过 O ( n l o g n ) O(nlogn) O(nlogn)
此外,该方法适用于动态数据流的使用场景。在不断加入数据时,我们可以持续维护堆内的元素,从而实现最大的 k k k个元素的动态更新。

代码如下:

vector<int> topKHeap(vector<int> &nums, int k) {
	//初始化一个最小堆
	priority_queue<int, vector<int>, gereater<int>> heap;
	//先将前k个元素入堆
	for (int i = 0; i < k; i++) {
		heap.push(nums[i]);
	}
	//遍历整个数组,维护大小为k的小顶堆
	for (int i = k; i < nums.size(); i++) {
		//若当前元素大于堆顶元素,则将堆顶元素出堆、当前元素入堆
		if (nums[i] > heap.top()) {
			heap.pop();
			heap.push(nums[i]);
		}
	}
	//将堆中的元素收集到结果像两种
	vector<int> result;
	while (!heap.empty()) {
		result.push_back(heap.top());
		heap.pop();
	}
	return result;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

JS 实战 贪吃蛇游戏

一、css 部分 1. 居中 想要开始和暂停两个按钮居中&#xff0c;可以将盒子设置为弹性盒 也可以使用其他方法 【代码】 2. 将父元素设置为相对定位&#xff0c;偏于之后贪吃蛇长长的身子&#xff0c;是以父元素为基点的绝对定位&#xff0c;通过 left 和 top 来控制位置 二、…

vue表格中上传按钮样式

问题&#xff1a;写了样式但是遇到问题如下图&#xff1a; 解决方法&#xff1a; ::v-deep .el-upload {display: flex;justify-content: center;align-items: center; } 因为上传的图标被包含在el-upload中&#xff0c;而删除按钮并没有被包含在el-upload中。 所以整体的样式…

存储+调优:存储-IP-SAN

存储调优&#xff1a;存储-IP-SAN 数据一致性问题 硬盘&#xff08;本地&#xff0c;远程同步rsync&#xff09; 存储设备&#xff08;网络&#xff09; 网络存储 不同接口的磁盘 1.速率 2.支持连接更多设备 3.支持热拔插 存储设备什么互联 千…

ACM实训

【碎碎念】继续搞习题学习&#xff0c;今天完成第四套的ABCD&#xff0c;为下一周挤出时间复习&#xff0c;加油 Digit Counting 问题 法希姆喜欢解决数学问题。但有时解决所有的数学问题对他来说是一个挑战。所以有时候他会为了解决数学难题而生气。他拿起一支粉笔&#xff…

岛屿问题刷题

200. 岛屿数量 - 力扣&#xff08;LeetCode&#xff09; class Solution {public int numIslands(char[][] grid) {int n grid.length;//grid行数int m grid[0].length;//grid列数int res 0;for(int r 0;r<n;r){for(int c0;c<m;c){if(grid[r][c]1){dfs(grid,r,c);res…

HCIP-VLAN综合实验

一、实验拓扑 二、实验要求 1、pc1和pc3所在接口为access;属于vlan 2; PC2/PC4/PC5/PC6处于同一网段’其中PC2可以访问PC4/PC5/PC6; PC4可以访问PC6&#xff1b;PC5不能访问PC6&#xff1b; 2、PC1/PC3与PC2/PC4/PC5/PC6不在同一个网段&#xff1b; 3、所有PC通过DHCP获取IP…

Multi-Attention Transformer for Naturalistic Driving Action Recognition

标题&#xff1a;用于自然驾驶行为识别的多注意力Transformer 源文链接&#xff1a;https://openaccess.thecvf.com/content/CVPR2023W/AICity/papers/Dong_Multi-Attention_Transformer_for_Naturalistic_Driving_Action_Recognition_CVPRW_2023_paper.pdfhttps://openaccess…

安装ollama并部署大模型并测试

Ollama介绍 项目地址&#xff1a;ollama 官网地址&#xff1a; https://ollama.com 模型仓库&#xff1a;https://ollama.com/library API接口&#xff1a;api接口 Ollama 是一个基于 Go 语言开发的简单易用的本地大语言模型运行框架。可以将其类比为 docker&#xff08;同基…

鸿蒙ArkUI-X跨平台技术:【SDK结构介绍】

ArkUI-X SDK目录结构介绍 简介 本文档配套ArkUI-X&#xff0c;将OpenHarmony ArkUI开发框架扩展到不同的OS平台&#xff0c;比如Android和iOS平台&#xff0c;让开发者基于ArkUI&#xff0c;可复用大部分的应用代码&#xff08;UI以及主要应用逻辑&#xff09;并可以部署到相…

深度学习之人脸性别年龄检测系统

欢迎大家点赞、收藏、关注、评论啦 &#xff0c;由于篇幅有限&#xff0c;只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 一、项目背景与意义 随着计算机视觉和深度学习技术的飞速发展&#xff0c;人脸性别年龄检测系统在多个领域展现出广…

简易Docker磁盘使用面板Doku

这个项目似乎有 1 年多没更新了&#xff0c;最后发布版本的问题也没人修复&#xff0c;所以看看就行&#xff0c;不建议安装 什么是 Doku &#xff1f; Doku 是一个简单、轻量级的基于 Web 的应用程序&#xff0c;允许您以用户友好的方式监控 Docker 磁盘使用情况。Doku 显示 D…

【30天精通Prometheus:一站式监控实战指南】第6天:mysqld_exporter从入门到实战:安装、配置详解与生产环境搭建指南,超详细

亲爱的读者们&#x1f44b;   欢迎加入【30天精通Prometheus】专栏&#xff01;&#x1f4da; 在这里&#xff0c;我们将探索Prometheus的强大功能&#xff0c;并将其应用于实际监控中。这个专栏都将为你提供宝贵的实战经验。&#x1f680;   Prometheus是云原生和DevOps的…

JavaEE-网络初识

文章目录 一、网络背景1.1 起源1.2 国内网络的发展 二、关键概念2.1 网络2.2 设备2.3 ip地址与端口号 三、协议3.1 协议分层3.2 OSI七层模型3.3 TCP/IP五层模型3.4 数据传输过程的简单叙述 一、网络背景 1.1 起源 在国外大概时上世纪70年代左右&#xff0c;网络就出现了&…

鸿蒙OS开发:【一次开发,多端部署】(典型布局场景)

典型布局场景 虽然不同应用的页面千变万化&#xff0c;但对其进行拆分和分析&#xff0c;页面中的很多布局场景是相似的。本小节将介绍如何借助自适应布局、响应式布局以及常见的容器类组件&#xff0c;实现应用中的典型布局场景。 布局场景实现方案 开发前请熟悉鸿蒙开发指导…

与MySQL DDL 对比分析OceanBase DDL的实现

本文将简要介绍OceanBase的DDL实现方式&#xff0c;并通过与MySQL DDL实现的对比&#xff0c;帮助大家更加容易理解。 MySQL DDL 的算法 MySQL 的DDL实现算法主要有 copy、inplace和instant。 copy copy算法的实现相对简单&#xff0c;MySQL首先会创建一个临时表&#xff0…

服务器c盘爆满了,这几种方法可以帮助C盘“瘦身”

我们在使用服务器的时候基本不会在C盘安装软件&#xff0c;那么用久了发现C盘满了&#xff0c;提示空间不足&#xff1f;那么这是怎么回事&#xff0c;为什么空间会占用这么快呢&#xff1f; 原因一&#xff1a; C盘满了&#xff0c;很可能是因为电脑里的垃圾文件过多。操作系…

Servlet的request对象

request对象的继承关系 1.HttpServletRequest接口继承了ServletRequest接口&#xff0c;对其父接口进行了扩展&#xff0c;可以处理满足所有http协议的请求 2.HttpServletRequest和ServletRequest都是接口&#xff0c;不能创建对象&#xff0c;因此在tomcat底层定义实现类并创…

Google Find My Device:科技守护,安心无忧

在数字化的时代&#xff0c;我们的生活与各种智能设备紧密相连。而 Google Find My Device 便是一款为我们提供安心保障的实用工具。 一、Find My Decice Netword的定义 谷歌的Find My Device Netword旨在通过利用Android设备的众包网络的力量&#xff0c;帮助用户安全的定位所…

考场作弊行为自动抓拍分析系统

考场作弊行为自动抓拍分析系统采用了AI神经网络和深度学习算法&#xff0c;考场作弊行为自动抓拍分析系统通过人形检测和骨架勾勒等技术&#xff0c;实时计算判断考生的异常动作行为。通过肢体动作识别技术&#xff0c;系统可以详细分析考生的头部和手部肢体动作&#xff0c;进…

【oracle004】oracle内置函数手册总结(已更新)

1.熟悉、梳理、总结下oracle相关知识体系。 2.日常研发过程中使用较少&#xff0c;随着时间的推移&#xff0c;很快就忘得一干二净&#xff0c;所以梳理总结下&#xff0c;以备日常使用参考 3.欢迎批评指正&#xff0c;跪谢一键三连&#xff01; 总结源文件资源下载地址&#x…