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

目录

一、 二叉树的顺序结构

二、 堆的概念及结构

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

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

四、大小堆解释

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

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;
}

今天就先到这了!!!

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

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

分布式事务的解决方案--Seata架构

一、Seata的XA模式 二、AT模式原理 三、TCC模式原理 四、MQ分布式事务 异步&#xff0c;非实时&#xff0c;实现最终的一致性。 四、分布式事务的解决方案

uniapp+uview 学习笔记(二)—— H5开发

文章目录 前言一、开发步骤1.创建项目2.安装组件库并导入使用3.封装请求4.国际化5.打包 总结 前言 本文主要介绍使用uniapp框架和uview组件库进行H5开发&#xff0c;需要用到的开发工具为HBuilder X。 一、开发步骤 1.创建项目 打开HBuilder X&#xff0c;在顶部栏目选择 新…

python知识点总结(四)

这里写目录标题 1、Django 中的缓存是怎么用的&#xff1f;2、现有2元、3元、5元共三种面额的货币&#xff0c;如果需要找零99元&#xff0c;一共有多少种找零的方式?3、代码执行结果4、下面的代码执行结果为&#xff1a;5、说一下Python中变量的作用域。6、闭包7、python2与p…

使用华为云HECS服务器+nodejs开启web服务

简介: 在华为云HECS服务器上使用nodejs开启一个web服务。 1.开通华为云服务器 这里我已经开通过了。 2.远程登录 2.1 使用华为官方的网页工具登录 输入密码登录。这里的密码应该在创建服务器时设置过的&#xff0c;由于已经创建过了&#xff0c;所以无法演示。 成功登…

视频技术2:把rtsp转为各种格式,包括webrtc

前题是启动ABLMediaServer&#xff0c;把ini里的hls_enable1 1、添加rtsp到视频服务器 http://127.0.0.1:7088/index/api/addStreamProxy?secret035c73f7-bb6b-4889-a715-d9eb2d1925cc&vhost_defaultVhost_&appMedia&streamCamera_00001&enable_hls1&ur…

SQLiteC/C++接口详细介绍之sqlite3类(十八)

返回目录&#xff1a;SQLite—免费开源数据库系列文章目录 上一篇&#xff1a;SQLiteC/C接口详细介绍之sqlite3类&#xff08;十七&#xff09; 下一篇&#xff1a;SQLiteC/C接口详细介绍sqlite3_stmt类&#xff08;一&#xff09; ​ 56.sqlite3_update_hook 函数功能&am…

统计-R(相关系数)与R^2(决定系数)

1.相关系数&#xff08;R&#xff09; 定义&#xff1a;考察两个事物&#xff08;在数据里我们称之为变量&#xff09;之间的相关程度。 假设有两个变量X&#xff0c;Y&#xff0c;那么两个变量间的皮尔逊相关系数可通过以下公式计算&#xff1a; 公式一&#xff1a; 其中…

OkHttp

文章目录 OkHttp概要1.简介2.特点3.基本组成5.工作流程 拦截器1.简介2.内置拦截器3.自定义拦截器 连接池1.简介2.常用参数配置选项 Dispatcher和线程池1.简介2.重要方法3.DispatCher中的双端队列4.总结 OkHttp 概要 1.简介 OkHttp是一个开源的HTTP客户端&#xff0c;用于在J…

【Sass】1px分割线 + 缩进分割线

效果图 1. 亮色模式效果 2. 暗色模式效果 设计思路 配色使用grey色 优点&#xff1a;无论在暗色模式还是亮色模式都可以看清楚分割线 使用after,before 伪元素绘制线条&#xff0c;并压缩线条transform: scaleY(.25) 注意事项 必须确保父级有宽高父级定位必须为position: r…

uniapp+vue3+setup语法糖开发微信小程序时不能定义globalData的解决方法

在使用 uniapp 开发小程序的时候&#xff0c; 发现使用了setup 语法糖 &#xff0c;定义 globalData 时&#xff0c;要不是定义不了&#xff0c; 要不就是使用 getApp()取不到&#xff0c;后来想到一个不伦不类的方法解决了&#xff0c; 这个方法有点难看&#xff0c; 但是解决…

文件读取的高效方法与设计模式

⭐️ 导言 在软件开发中&#xff0c;经常需要处理各种类型的文件&#xff0c;包括文本文件&#xff08;如JSON、CSV、TXT&#xff09;、Excel 文件等。针对不同类型的文件&#xff0c;我们需要选择合适的方法来读取和处理文件内容。在本篇博客中&#xff0c;小编以python为例&…

【Windows 常用工具系列 15 -- VMWARE ubuntu 安装教程】

文章目录 安装教程镜像下载 工具安装 安装教程 安装教程参考链接&#xff1a;https://blog.csdn.net/Python_0011/article/details/131619864 https://linux.cn/article-15472-1.html 激活码 VMware 激活码连接&#xff1a;https://www.haozhuangji.com/xtjc/180037874.html…

工控MCGS触摸屏Hacking勒索部署

https://github.com/MartinxMax/Mo0n_V1.2 !!不关注点赞收藏,以后没好东西了奥!! 端口扫描-获取信息 $python Mo0n.py -scan x.x.x.0/24 or $nmap -sS -Pn -T4 x.x.x.0/24 -p 127 MCGS编程软件 发现触摸屏受到密码保护 Oops&#xff01;&#xff01;&#xff01; echo /\_…

基于深度学习YOLOv8+Pyqt5的工地安全帽头盔佩戴检测识别系统(源码+跑通说明文件)

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;318安全帽 获取完整源码源文件7000张已标注的数据集训练好的模型配置说明文件 可有偿59yuan一对一远程操作配置环境跑通程序 效果展示&#xff08;图片检测批量检测视频检测摄像头检测&#xff09; 基于深度学习YOLOv8Pyqt…

【Excel自动化办公】使用openpyxl对Excel进行读写操作

目录 一、环境安装 1.1 创建python项目 1.2 安装openpyxl依赖 二、Excel数据读取操作 三、Excel数据写入操作 3.1 创建空白工作簿 3.2 写数据 四、设置单元格样式 4.1 字体样式 4.2 设置单元格背景填充色 4.3 设置单元格边框样式 4.4 单元格对齐方式 4.5 数据筛选…

B008-springcloud alibaba 短信服务 sms

目录 短信服务介绍短信服务使用准备工作阿里云官网实名认证开通短信服务申请认证秘钥申请短信签名申请短信模板 短信服务API介绍短信发送(SendSms)短信查询(QuerySendDetails)功能测试 下单之后发送短信 短信服务介绍 短信服务&#xff08;Short Message Service&#xff09;是…

08-热点文章-定时计算-黑马头条

xxl-Job分布式任务调度 1 今日内容 1.1 需求分析 目前实现的思路&#xff1a;从数据库直接按照发布时间倒序查询 问题1&#xff1a; 如何访问量较大&#xff0c;直接查询数据库&#xff0c;压力较大 问题2&#xff1a; 新发布的文章会展示在前面&#xff0c;并不是热点文章 …

SpringCloud搭建微服务之Micrometer分布式链路追踪

1. 概述 由于Spring Cloud Sleuth最新版本只支持Spring Boot 2.7.x&#xff0c;核心项目已经迁移到Micrometer Traceing项目&#xff0c;Spring Boot 3.x版本要实现分布式链路追踪需要集成Micrometer。更多详情可以参阅Micrometer官网 本文将以Spring Boot 3.2.x和Spring Clo…

Cinema 4D 2024 for mac/Win:开启三维动画与建模新纪元

在数字化时代&#xff0c;三维动画与建模已成为影视、游戏、广告等多个领域不可或缺的创作工具。而Cinema 4D&#xff0c;作为这一领域的佼佼者&#xff0c;始终以其卓越的性能和创新的功能引领着行业的发展。如今&#xff0c;Cinema 4D 2024的发布&#xff0c;更是为我们带来了…

【command not found】原因分析及解决

在使用Linux时&#xff0c;会经常遇到 “command not found” 的错误。错误信息提示的是&#xff1a;Linux没有找到该命令。原因主要分类有&#xff1a; 1.命令拼写错误 2.软件路径配置错误 3.Linux 系统就没有安装该命令。 一、确认命令没有拼写错误 Linux 中的所有命令都是…