【数据结构】——堆的实现与算法

news2025/1/12 20:50:17

目录

一、堆的实现

1.1堆数据的插入

1.2堆数据的删除

二、建堆算法

2.1向上调整建堆

2.2向下调整建堆

三、堆的应用

3.1堆排序

3.2Top—K问题


一、堆的实现

1.1堆数据的插入

插入一个数据后不再是小堆需要将新数据调整到合适的位置,所以堆的插入就是在数组插入数据再向上调整即可

堆数据的插入

1.检查空间是否需要扩容

2.赋值size++

3.向上调整

void AdjustUp(HPDataType* a,int size)
{
	int child = size - 1;
	while (child > 0)
	{
		int parent = (child - 1) / 2;//父亲结点
		if (a[child] < a[parent])//判断 小于建小堆
		{
			Swap(&a[child], &a[parent]);
			child = parent;
		}
		else
		{
			break;
		}
	}
}
void HPPush(HP* php, HPDataType x)
{
	assert(php);
	if (php->capacity == php->size)
	{
		int newcapacity = php->capacity == 0 ? 4 : 2 * php->capacity;
		HPDataType* tmp = (HPDataType*)realloc(php->a,sizeof(HPDataType)*newcapacity);//这是 php->a的动态内存
		if (tmp == NULL)
		{
			perror("fail realloc");
			return;
		}
		php->a = tmp;
		php->capacity = newcapacity;
	}
	php->a[php->size++] = x;
	AdjustUp(php->a, php->size);
}

1.2堆数据的删除

分析:

因此:

代码:

void AdjustDown(HPDataType* a, int size, int parent)
{
	int child = 2 * parent + 1;
	while (child<size)//结束条件不能越界
	{
		//假设法求出最小的孩子
		if (child + 1 < size && a[child] > a[child + 1])
		{
			child = child + 1;
		}
		//调整
		if (a[parent] > a[child])
		{
			Swap(&a[parent], &a[child]);
			//更新父亲孩子结点
			parent = child;
			child = 2 * parent + 1;
		}
		else
		{
			break;
		}
	}
}
void HPPop(HP* php)//删除根结点
{
	assert(php && php->size > 0);
	Swap(&php->a[0], &php->a[php->size - 1]);
	php->size--;
	AdjustDown(php->a, php->size, 0);
}

二、建堆算法

2.1向上调整建堆

堆插入的过程其实就是建堆,在一个堆的末尾插入数据然后向上调整形成一个新的堆

void AdjustUp(HPDataType* a, int size)
{
	int child = size - 1;
	while (child > 0)
	{
		int parent = (child - 1) / 2;//父亲结点
		if (a[child] > a[parent])//判断 小于建小堆
		{
			Swap(&a[child], &a[parent]);
			child = parent;
		}
		else
		{
			break;
		}
	}
}

向上调整建堆时间复杂度分析

根据大O渐进法,向上调整建堆的时间复杂度即 O(N*logN)

2.2向下调整建堆

从最后一个父亲结点开始依次向下调整

根据大O渐进法,向下调整建堆的时间复杂度即 O(N)

对比向上调整建堆,h-1层向下调整只需要移动1层,而向上调整需要移动h-1次因此向下调整是更优的建堆算法

三、堆的应用

3.1堆排序

堆排序即利用堆的思想来进行排序,总共分为两个步骤:

1. 建堆

升序:建大堆

降序:建小堆

2. 利用堆删除思想来进行排序 

演示代码:

//升序向下调整
void SAdjustDown(HPDataType* a, int size, int parent)
{
	int child = 2 * parent + 1;
	while (child < size)//结束条件不能越界
	{
		//假设法求出最小的孩子
		if (child + 1 < size && a[child] < a[child + 1])
		{
			child = child + 1;
		}
		//调整
		if (a[parent] < a[child])
		{
			Swap(&a[parent], &a[child]);
			//更新父亲孩子结点
			parent = child;
			child = 2 * parent + 1;
		}
		else
		{
			break;
		}
	}
}

void HeapSort(int* a, int n)
{
	//升序建大堆
	//降序建小堆

	向上建堆
	//for (int i = 1; i < n; i++)
	//{
	//	//相当于慢慢插入数据建堆
	//	SAdjustUp(a, i);
	//}
	//向下调整建堆
	for (int i = (n - 1 - 1) / 2; i >= 0; i--)
	{
		SAdjustDown(a, n,  i);
	}
	int end = n - 1;
	while (end > 0)
	{
		Swap(&a[0], &a[end]);
		SAdjustDown(a, end, 0);
		end--;
	}
}

 堆排序的时间复杂度:

对比向上调整建堆

可以知道堆排序时间复杂度:O(N*logN)

3.2Top—K问题

Top—K问题:N个数中求最大的(最小的)前K个数据,一般N远大于K

实例:专业前5名、富豪榜、游戏战力榜等等

方法一:

建一个N个数的大堆  O(N)

Pop k次数据         O(N*logN)

此方法缺陷也很明显如果N太大,排序就不太可取,甚至数据都无法一下子加载到内存中

方法二:

用前K个数建一个小堆   O(K)

剩下数据与堆顶数据比较,如果比堆顶数据大,那么替代堆顶进堆(覆盖堆顶数据然后向下调整)

O(logK*(N-K)(最坏情况都需要覆盖调整)

复杂度:O(N)

分析:

剩下的N-K个数据依次与堆顶数据比较

<1>数据比堆顶数据小,那么不可能为最大的前K个

<2>数据比堆顶数据大,覆盖调整之后形成一个新的小堆,再次比较

比较完所有的数后那么小堆中就是最大的前K个数据

代码实现

先生成十万个数据

void CreatData()
{
	int n = 100000;//生成10万个数据
	srand(time(0));//生成不同的种子
	FILE* file = fopen("qsy.txt", "w");//打开文件
	for (int i = 0; i < n; i++)
	{
		int x = rand() % 100001 + i;//生成随机数
		fprintf(file, "%d\n", x);//写数据
	}
	fclose(file);//关闭文件
	file = NULL;
}

开K个空间从 qsy 文件里读取数据建堆

之后再读取数据比较堆顶数据覆盖调整

void CreatData()
{
	int n = 100000;//生成10万个数据
	srand(time(0));//生成不同的种子
	FILE* file = fopen("qsy.txt", "w");//打开文件
	for (int i = 0; i < n; i++)
	{
		int x = rand() % 100001 + i;//生成随机数
		fprintf(file, "%d\n", x);//写数据
	}
	fclose(file);//关闭文件
	file = NULL;
}

int main()
{
	CreatData();
	int k;
	scanf("%d", &k);
	int* kmin = (int*)malloc(10 * sizeof(int));
	if (kmin == NULL)
	{
		perror("malloc fail");
		return 1;
	}
	//读取数据
	FILE* pf = fopen("qsy.txt", "r");
	for (int i = 0; i < k; i++)
	{
		fscanf(pf, "%d", &kmin[i]);
	}
	//建k个数的小堆
	for (int i = (k - 1 - 1) / 2; i >= 0; i--)
	{
		JAdjustDown(kmin, k, i);
	}
	//比较数据
	int x;
	while (fscanf(pf, "%d", &x) != EOF)
	{
		if (x > kmin[0])
		{
			kmin[0] = x;
			JAdjustDown(kmin, k, 0);
		}
	}
	//打印数据
	for (int i = 0; i < k; i++)
	{
		printf("%d ", kmin[i]);
	}
	return 0;
}

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

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

相关文章

永结无间Ⅷ--释放机器学习“百万专家组合”的力量

介绍 随着人们对特定领域模型和小型语言模型的兴趣日益浓厚&#xff0c;对于能够利用专门数据有效处理特定领域中大量专门任务请求的模型的需求比以往任何时候都更加迫切。 传统模型往往难以实现可扩展性和适应性&#xff0c;因此研究人员开始探索新的创新方法。DeepMind的“…

【课程系列10】某乎AI大模型全栈工程师-第5期

网盘链接 链接&#xff1a;https://pan.baidu.com/s/1ZC-fOC_QQjNM6wyVjQcYOg --来自百度网盘超级会员v6的分享 课程目标 ✅学大模型来干什么&#xff1f;&#xff1a; &#x1f449;想提升薪资&#xff0c;提升效率 &#x1f449;想私有化部署垂直领域大模型 &#x1f449…

误删文件怎么恢复?电脑误删文件恢复,5个方法深度解析(超实用)

在日常使用电脑的过程中&#xff0c;相信很多小伙伴们都曾经遇到过一个普遍又棘手的问题——误删文件。无论是因为手快便捷还是不慎操作&#xff0c;误删文件似乎总是那么糟心&#xff0c;让人十分困扰。然而&#xff0c;当我们面对这一问题时&#xff0c;我们并不必过分担心。…

mysql+php+html实现学生管理系统

mysqlphphtml实现学生管理系统 前言 本文使用Mysqlphphtml实现一个简单的学生管理系统&#xff0c;实现了登陆&#xff0c;注册&#xff0c;总览学生信息&#xff0c;添加学生&#xff0c;查询特定的学生&#xff0c;删除指定的学生等功能。并且本文仅用来学习就够了&#xf…

大数据——Hive原理

摘要 Apache Hive 是一个基于 Hadoop 分布式文件系统 (HDFS) 的数据仓库软件项目&#xff0c;专为存储和处理大规模数据集而设计。它提供类似 SQL 的查询语言 HiveQL&#xff0c;使用户能够轻松编写复杂的查询和分析任务&#xff0c;而无需深入了解 Hadoop 的底层实现。 Hive…

【pikachu】文件上传漏洞 第三关getimagesize

思路&#xff1a;上传图片马&#xff0c;利用同服务器下的文件包含漏洞解析图片马 对文件后缀名进行白名单检验 对上传包进行修改&#xff1a; 图片后缀图片MIMEGIF89a一句话木马 上传成功 但是图片无法当作php解析 来到文件包含漏洞页面&#xff0c;尝试将刚才上传的图片马…

32--新建工程

一、keil编译完成之后&#xff0c;下载时弹出unknown target connected错误提示 调一下下面&#xff1a; 二、点亮 1&#xff1a;配置RCC&#xff0c;使能&#xff0c;时钟 2&#xff1a;GPIOC高寄存器 3&#xff1a;给端口数据 三、ST-LINK USB communication error 非常有…

uniapp中实现语音识别(app+小程序)

一.app版本需要先去百度智能云申请 注意填写完&#xff0c;需要打包成自定义基座或者安装rpk包&#xff0c;本地是无效的封装recording-popup.vue组件 <template><up-popup round"16" closeable :show"recordShow" :close-on-click-overlay&qu…

计算机网络—电路、分组、报文交换—图文详解

计算机网络—电路、分组、报文交换 计算机网络中的数据传输方式可以根据数据的处理方式和网络资源的使用方式分为电路交换、分组交换和报文交换三种类型。 这些方式在网络设计和数据传输过程中起到了不同的作用和效果。 1. 电路交换&#xff08;Circuit Switching&#xff0…

数字中国:智能交通的未来发展方向

随着数字中国的不断推进&#xff0c;智能交通作为数字化时代的一个重要领域&#xff0c;正面临着前所未有的机遇和挑战。人工智能、大数据应用和物联网等新兴技术的加入&#xff0c;不仅改变了传统交通的运行模式&#xff0c;还赋予了智能交通更多的功能和价值。首先&#xff0…

Convert Ensembl IDs to gene symbols python包

links&#xff1a; https://pypi.org/project/ensembl-converter/ pip install Ensembl_converter批量转&#xff1a; from Ensembl_converter import EnsemblConverter# Create an instance of EnsemblConverter converter EnsemblConverter()# Provide a list of Ensembl …

IEC MMS协议源码运行

环境准备 源码下载链接 https://github.com/mz-automation/libiec61850 我的运行环境是ubuntu虚拟机。 首先进入文件夹根目录进行编译 make clean makeserver代码的编译和运行 进入examples/serve_example_simpler的目录下编译 直接执行会发现报错&#xff0c;异常退出。 …

JLink烧录失败

1. 现象&#xff1a; 这个位置是灰色的&#xff0c;没有SW Device信息。 MDK下面的打印&#xff1a; J-Flash的打印&#xff1a; windows上面的弹框的现象没有截屏。 2. 解决办法&#xff1a; 1.打开J-Link Commander,输入unlock kinetis&#xff0c;看现象不起作用,网…

Python学习笔记48:游戏篇之外星人入侵(九)

前言 到目前为止&#xff0c;飞船&#xff0c;子弹&#xff0c;外星人的创建&#xff0c;移动都已经完成。接下来我们需要完成功能主要就是子弹击中外星人和飞船接触到外星人两个中功能。 碰撞 我们需要实现的功能中&#xff0c;子弹击中外星人和飞船接触外星人本质上就是两…

Navicat For Mysql连接Mysql8.0报错:客户端不支持服务器请求的身份验证协议

windows通过navicat连接本地mysql时报错:Client does not support authentication protocol requested by server; consider upgrading MySQL client 一、问题原因二、解决方法1--失败1. 连接mysql客户端2. 修改加密方式3.正确的解决方法1.查找my.ini文件2.修改my.ini文件3.重…

【读点论文】Object Detection in 20 Years: A Survey,宏观了解大方向发展,常看常新,这篇越看到后面越泛

Object Detection in 20 Years: A Survey Abstract 目标检测作为计算机视觉领域最基本、最具挑战性的问题之一&#xff0c;近年来受到了极大的关注。在过去的二十年里&#xff0c;我们目睹了目标检测技术的快速演进及其对整个计算机视觉领域的深远影响。如果说今天的目标检测…

关于RAG进展|| RankRAG:在大模型中统一检索增强生成的上下文排序

欢迎了解公众号&#xff1a;AI论文解读 背景&#xff1a;探索RankRAG的创新之处 检索增强生成&#xff08;Retrieval-Augmented Generation, RAG&#xff09;技术已成为提升大型语言模型&#xff08;Large Language Models, LLMs&#xff09;处理知识密集型任务的关键方法。传…

使用AI大模型Kimi轻松助力速通代理IP知识

本文目录 一、 引言二、代理IP介绍2.1 代理IP定义2.2 代理IP的工作原理2.3 代理IP的分类2.4 2.4 为什么需要代理IP&#xff1f; 三、代理IP的使用场景四、如何选择合适的代理IP服务五、使用代理IP的基本步骤六、使用代理IP爬取亚马逊电商信息七、总结八、代码附录 一、 引言 喜…

怎么使用rdma-core进行调用开发?

RDMA (Remote Direct Memory Access) 是一种网络协议,可以在计算节点之间实现高效的内存数据传输,而无需CPU的干预。rdma-core 是 RDMA 的一个用户空间库,提供了一些简单易用的接口来使用 RDMA 功能。 目录: 一、环境准备: 1.1 安装依赖 在安装 rdma-core 之前,确保你的…

token验证

验证客户端传输过来的请求是否合法 try-catch是用来捕获并处理异常的。当你在编写代码时&#xff0c;可能会遇到一些不可预见的情况&#xff0c;这些情况会阻止代码的正常执行&#xff0c;这时就会抛出异常。使用try-catch语句&#xff0c;你可以捕获这些异常并采取相应的措施来…