堆结构知识点复习——玩转堆结构

        前言:堆算是一种相对简单的数据结构, 本篇文章将详细的讲解堆中的知识点, 包括那些我们第一次学习堆的时候容易忽略的内容, 本篇文章会作为重点详细提到。 

        本篇内容适合已经学完C语言数组和函数部分的友友们观看。

        

目录

什么是堆

建堆算法

向上调整算法

算法原理

如何计算parent

代码

向下调整算法

算法原理

寻找较小孩子 

代码

建堆

向下调整算法建堆

建堆过程

建堆的时间复杂度

向上调整算法建堆

建堆过程

建堆的时间复杂度


什么是堆

        首先来看什么是堆:

  • 堆在逻辑结构上是一种完全二叉树。
  • 堆的物理结构是数组。
  • 堆分为大根堆和小根堆。
  • 大根堆就是父亲节点大于左右孩子, 小根堆就是父亲节点小于左右父亲。

这里分析一个问题:堆相较于顺序表存不存在大量的空间浪费?

普通的二叉树如果使用数组来存储会出现大量的空间浪费。 但是堆是一颗完全二叉树, 完全二叉树就是除了叶子结点, 其他层上面的分支节点全都是满的。 而且叶子结点也全部是连续的, 这样的结构如果使用数组来存储就不会存在大量浪费的情况。 

逻辑结构的堆:

物理结构的堆:

建堆算法

建堆有两个算法:

  • 向上调整算法
  • 向下调整算法

向上调整算法

向上调整算法是从堆底向上调整。

向上调整算法的使用前提是:要向上调整的节点的前面的数组已经是一个堆。

算法原理

示例:

如图为一个小堆:

        1.现在插入一个0。那么这个0先插入在最后的位置。

         然后向上调整算法的过程就是:2.以刚刚插入的位置为child节点。 他的父亲为parent节点。 进行比较。

        3.比较child比parent小(注意, 现在是排的小堆, 小堆的父亲比左右孩子小), 那么就要向上调整一下, 让父亲变成0, child变成3。(这里如果不小的话, 就说明此时就是一个小堆, 那么就结束调整。结束算法)

        4.然后让child变成指向parent的节点。 parent指向当前节点的父亲节点。 

       5. 然后回到2, 重新循环.  直到遇到child不小于parent或者child已经是堆顶元素。如图:

此时child指向堆顶。 没有父亲节点, 也就不需要再进行向上调整。 

时间复杂度:

  • O(lgN)。

因为每次调整一下最坏的情况就是从堆底调整到堆顶。 对于一颗满二叉树(没写错, 就是满二叉树)来说, 有2 * h - 1 == N;   一棵满二叉树的高度是h == lg(N + 1)。所以它向上调整一次最坏的情况是调整lgN次(1被忽略)。那么对于完全二叉树来说, 比满二叉树还要少许多节点, 层数一样。 那么它的最坏调整次数也是lgN。所以时间复杂度就是lgN。 

如何计算parent

        假设有一个元素个数为6的小堆。
 

此时元素个数为6。 堆的物理结构就是:

从逻辑结构中我们可以看到对于3这个位置来说, 它的左右两个孩子的下标是5和6。 (5 - 1) / 2 == 2 ; (6 - 1) /   2 == 2;

对于图中的5节点来说,5的下标是1, 它的左右孩子的下标是3和4。 而 (3 - 1) / 2 == 1; (4 - 1)  / 2 == 1;

这里就有一个结论:

  • parent == (child - 1) / 2;

代码

//向上调正算法
void AdjustUp(int* arr, int n)   //arr是一个等待调整的数组, n是这个数组的大小。
{//要使用向上调整算法, 说明此时arr的前n - 1个元素是一个堆。 第n个元素是等待向上调整的元素。

	int child = n - 1;              //第n个元素的下标是n - 1。     
	int parent = (child - 1) / 2;   //算出parent的位置
	while (child > 0)               //如果child到了堆顶, 那么就没有父亲, 也就不用比了。
	{
		if (arr[child] < arr[parent]) 
		{
			swap(&arr[child], &arr[parent]);
			child = parent;
			parent = (parent - 1) / 2;

		}
		else                        //如果孩子节点比父亲节点还大, 那么也不用比了, 此时就是个堆
		{
			break;
		}

	}
}

向下调整算法

算法原理

向下调整算法要求左右两棵子树都是堆, 然后以根节点为基准向下进行调整。 

示例:

        如图是一个数组

这个数组如果转化为堆的逻辑结构就是如图:

        观察逻辑结构, 我们很容易可以观察到9的左右两边都是小堆。符合向下调整算法的要求(如果左右两边有一边不是堆结构, 那么就不可以使用向下调整算法),向下调整算法的流程为:

        1.  定义parent指向9的位置。 child指向左右边小的那个节点。 因为这里1比5小, 所以指向1。

          2.  然后比较child和parent。 如果child小于parent。 那么就要交换位置, 否则退出循环, 算法结束。

        3.  交换数据之后移动指针, parent指向child指向的节点。 child指向当前节点的左右孩子中较小的那个。

        4.  然后重复2的操作。 一直到child大于parent或者查出数组的范围。退出循环, 算法结束.

时间复杂度

  • O(lgN)

       时间复杂度的计算和向上调整算法一样。 对于一个堆来说,最坏的情况就是从堆顶向下调整到堆底,那么调整次数就是树的高度次。 而树的高度的数量级是lgN级别。 所以时间复杂度就是O(lgN)。

寻找较小孩子 

        可以利用假设法寻找左右孩子中较小的那一个。       

        对于一个父亲节点, 它的左孩子的下标是child == parent * 2 + 1; 右孩子就是child == parent * 2 + 2;

假设过程如下:

  •      先假设左孩子是较小的那个孩子。
  •      然后, 就比较左孩子和右孩子
  •      如果右孩子比左孩子更小, 那么就让右孩子变成较小的那个孩子。

代码


//向下调整算法
void AdjustDown(int* arr, int n, int parent)  //arr是要调整的数组, n是要建堆数组的大小。 parent下调的基准点。
{
	int child = parent * 2 + 1;               //先假设孩子节点是父亲节点的左边

	while (child < n)                         
	{
		//如果右孩子更小, 那么就让child变成父亲节点的右边
		if (child + 1 < n && arr[child + 1] < arr[child]) child++;              
		if (arr[child] < arr[parent]) 
		{
			swap(&arr[child], &arr[parent]);

			parent = child;
			child = child * 2 + 1;
		}
		else 
		{
			break;
		}
	}
}

建堆

向下调整算法建堆

建堆过程

        向下调整建堆需要保证那个要调整的节点的左右子树都是堆。 所以我们进行向下调整建堆的时候要从堆底向堆顶建堆。 具体过程如下:

假设如图为要调整成为堆的数组的逻辑结构:

       我们首先要从堆底的第一个非叶子节点开始向下调整,就像下图的最右边的那个红框框。 从这个红框框中的非叶子节点开始向下调整。 从右向左, 先将这一层的非也节点全部调整为堆。 

       此时, 这一层往下都是堆结构, 那么我们就可以向上一层进行调堆。

 

 当我们调好红框框中的堆后, 就可以调绿框框的堆。 

 然后绿框框的堆调好之后我们就可以调以堆顶为基准的堆, 那么这整个数组就建堆完成。 

 

建堆的时间复杂度

        个人认为堆里面最难的一部分内容, 就是建堆的时间复杂度。 可能有的友友会说, 建堆的时间复杂度是O(N * lgN) 。 博主一开始也以为是O(N * lg N), 但是其实只有向上调整建堆是O(N * lgN)。 而向下调整建堆的时间复杂度其实是O(N)。 为什么? 这里其实用到了高中的错位相减法求时间复杂度。

        假设这是一个h层的树结构。

那么当我们建堆的时候。  从倒数第二层开始向下调整。假设一共h层。证明如下:

  • 第h - 1层有2 ^ (h - 2) 个节点, 每一个节点最多向下调整一次。 一共调整2 ^ (h - 2) * 1次。
  • 倒h - 2层有2 ^ (h - 3) 个节点, 每一个节点最多向下调整两次。 一共调整2 ^ (h - 3) * 2次。
  • 倒h - 3层有2 ^ (h - 4) 个节点, 每一个节点最多向下调整三次。 一共调整2 ^ (h - 4) * 3次。
  • …………
  • …………
  • …………
  • 第三层有2 ^ 2个节点, 每一个节点最多向下调整h - 3次。一共调整 2 ^ 2 * (h - 3)次。
  • 第二层有2 ^ 1个节点, 每一个节点最多向下调整h - 2次。 一共调整2 ^ 1 * (h - 2)次。
  • 第一层有2 ^ 0个节点, 每一个节点最多向下调整h - 1次。 一共调整2 ^ 0 * (h - 1)次。

这些次数加起来就是一个等差乘以等比类型的数列求和。

        T(h) == 2 ^ 0 * (h - 1) + 2 ^ 1 * (h - 2) + 2 ^ 2 * (h - 3) + …… + 2 ^ (h - 4) * 3 + 2 ^ (h - 3) * 2 + 2 ^ (h - 2) * 1

   2 * T(h) ==             2 ^ 1 * (h - 1) + 2 ^ 2 * (h - 2) + …… + 2 ^ (h - 4) * 4 + 2 ^ (h - 3) * 3 + 2 ^ (h - 2) * 2 + 2 ^ (h - 1) * 1;

        所以 :T(h) == - (h - 1) + 2 + 2 ^ 2 + 2 ^ 3 + …… + 2 ^ (h - 2) + 2 ^ (h - 1)

                           == 2 ^ h - h;

由完全二叉树的规则: N == 2 ^ h - 1; 将这个式子带入上面T(h)就能得到T(N) == N + 1 - lgN

由大O的渐进表示法可知, 时间复杂度为O(N)

代码:


void CreatHeap(int* arr, int sz) 
{
	for (int i = (sz - 1 - 1) / 2; i >= 0; i--)   //(sz - 1) / 2是第一个非叶节点。
	{
		//向下调整建堆
		AdjustDown(arr, sz, i);    //以i为基准, 最大下标为sz;
	}
}

向上调整算法建堆

建堆过程

向上调整建堆需要前面的数组为堆结构。 所以和向下调整建堆相反,它是从堆顶开始建堆。建堆过程需要从下标为0开始向后遍历进行向上调整建堆。 建堆过程如下:

如图为一树结构。 

对于这个结构。 我们要先从第一个堆顶位置开始向下调。

然后调第二个元素

再调第三个元素, 第四个元素……第n个元素。 依次类推。

建堆的时间复杂度

        我们同样使用前面的方法, 将每一层的节点的调整次数列出来。 如下:

  • 第一层:2 ^ 0节点, 调整0次
  • 第二层:2 ^ 1节点, 每个节点调整1次。一共调整2 ^ 1 * 1次。
  • 第三层:2 ^ 2节点, 每个节点调整2次。一共调整2 ^ 2 * 2次。
  • 第四层:2 ^ 3节点, 每个节点调整3次。 一共调整2 ^ 3 * 3次。
  • …………
  • …………
  • …………
  • 第h - 2层: 2 ^ (h - 3)节点, 每个节点调整h - 3次。 一共调整2 ^ (h - 3) * (h - 3)次。
  • 第h - 1层:2 ^ (h - 2)节点, 每个节点调整h -2次。 一共调整2 ^ (h - 2) * (h - 2)次。
  • 第h 层: 2 ^ (h - 1)节点, 每个节点调整h - 1次。 一共调整2 ^ (h - 1) * (h - 1)次。

       利用N == 2 ^h - 1代入可得最后一层节点个数大约为N / 2。(这其实也是满二叉树的性质。)

       所以我们只需要看第h层调整需要的节点个数, 因为对于一棵完全二叉树来说, 最后一层的节点个数相当于其所有节点个数的一半。 那么每个节点向上调整的次数都是lgN。 所以对于向上调整算法建堆的时间复杂度就是O(N * lgN)

代码:

void CreatHeap(int* arr, int sz) 
{
	for (int i = 0; i < sz; i++)  
	{
		//向上调整建堆
		AdjustUp(arr, i);    //以i为调堆最后一个元素
	}
}

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

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

相关文章

蓝桥杯2023(十四届)省赛——接龙数列(DP)

接龙数列&#xff08;DP&#xff09; 1.接龙数列 - 蓝桥云课 (lanqiao.cn) 琢磨半天&#xff0c;本来是开一个三维的&#xff0c;dp[i][j][k] 表示 前i个&#xff0c;以j为首项&#xff0c;k为尾项的最大子集个数&#xff0c;但是实际上用二维即可。想求的是删除个数&#xf…

java并发工具类都有哪些

Java中的并发工具类包括&#xff1a; CountDownLatch CountDownLatch允许一个或多个线程等待其他线程完成某些操作。它通常用于线程间的同步&#xff0c;例如在一个线程完成其工作后通知其他线程继续执行。 CyclicBarrier CyclicBarrier是一个同步辅助类&#xff0c;它允许一…

STM32H743+USBHID+CubeMX配置

一、环境准备 电脑系统&#xff1a;Windows 10 专业版 20H2 IDE&#xff1a;Keil v5.35、STM32CubeMX v6.5.0 测试硬件&#xff1a;正点原子阿波罗STM32H743 二、测试步骤 1、使用用例工程 配置STM32H743定时器功能-CSDN博客https://blog.csdn.net/horse_2007s/article/d…

基于51单片机的电压表-数码管显示

一.硬件方案 本设计基于STC89C52单片机的一种电压测量电路&#xff0c;该电路采用ADC0832A/D转换芯片,实现数字电压表的硬件电路与软件设计。该系统的数字电压表电路简单, 可以测量0&#xff5e;9V的电压值,并在四位LED数码管上显示电压值。 二.设计功能 &#xff08;1&…

HNCTF

HNCTF 文章目录 HNCTFBabyPQEZmathez_Classicf(?*?)MatrixRSABabyAESIs this Iso? BabyPQ nc签到题&#xff0c;跟端口连接拿到n和phin n 8336450100232098099043686671148282601664696810002345240872579498695511770993195704402414029892029461830476866385453475141207…

211大学计算机专业不考408,新增的交叉专业却考408!南京农业大学计算机考研考情分析!

南京农业大学信息科技学院可追溯至1981年成立的计算中心和1985年筹建的农业图书情报专业。1987年设立了农业图书情报系&#xff0c;1993 年农业图书情报系更名为信息管理系&#xff0c;本科专业名称也于1999年更名为信息管理与信息系统专业。1994年计算中心开始招收计算机应用专…

SpringBootTest测试框架四

dubbo调用mock 同理,为了实现dubbo的mock,也是要在dubbo调用的过程中添加拦截器 dubbo原始的执行拦截器 Proxy.getProxy(interfaces).newInstance(new InvokerInvocationHandler(invoker)) 这里代码写死了,没办法了,只能将整个JavassistProxyFactory 替换掉 public class J…

Nvidia Orin/Jetson +GMSL/RLINC/VbyOne/FPDLink 同轴AI多相机同步车载视觉解决方案

在本次演讲中&#xff0c;介绍了多相机同步技术在自主机器中的应用情况&#xff0c;围绕无人配送小车、控制器视觉传感器方案升级、人形机器人三个典型案例中如何为客户提供高效的多相机同步解决方案进行了详细的讲解&#xff0c;并进一步介绍如何通过创新的多相机同步技术&…

Facebook的心灵之镜:探寻数字社交的灵魂深处

在当今数字化时代&#xff0c;社交媒体已经成为了人们生活的一部分&#xff0c;而Facebook作为其中的佼佼者&#xff0c;更是承载了数以亿计的用户情感和交流。然而&#xff0c;Facebook不仅仅是一个简单的社交平台&#xff0c;它更像是一面心灵之镜&#xff0c;反映着数字社交…

TS(TypeScript)中Array数组无法调出使用includes方法,显示红色警告

解决方法 打开tsconfig.json文件&#xff0c;添加"lib": ["es7", "dom"]即可。 如下图所示。

基于Qt的网上购物系统的设计与实现

企鹅&#xff1a;2583550535 代码和论文都有 第1章 绪论... 1 1.1 项目背景... 1 1.2 国内外研究现状... 1 1.3 项目开发意义... 3 1.4 报告主要内容... 3 第2章 关键技术介绍... 4 2.1 后端开发技术... 4 2.1.1 C. 4 2.1.2 Qt框架... 4 2.1.3 MySQL数据库... 5 2.2 …

Java开发大厂面试第20讲:什么是分布式锁?Redi 怎样实现的分布式锁?

“锁”是我们实际工作和面试中无法避开的话题之一&#xff0c;正确使用锁可以保证高并发环境下程序的正确执行&#xff0c;也就是说只有使用锁才能保证多人同时访问时程序不会出现问题。 我们本课时的面试题是&#xff0c;什么是分布式锁&#xff1f;如何实现分布式锁&#xf…

DSP开发入门

视频&#xff1a; 创龙TI 最新DSP CPU核心架构 C66x 以及 KeyStone I 架构 DSP TMS320C6655/57以及TMS320C6678视频教程全集_哔哩哔哩_bilibili 2024年硬汉科技手把手教您学DSP28335视频教程持续更新中_哔哩哔哩_bilibili DSP芯片介绍 DSP选型 TI的DSP 分为三大系列&#…

comfyui电商场景工作流总结

eSheep(内测中) - 一站式的AIGC社区eSheep.com 是国内知名的AIGC在线画图网站,提供海量模型,并支持在线AI画图。用户会上传自己的AIGC作品到网站上,进行交流。eSheep让AIGC更轻松,让更多人在AIGC中找到快乐https://www.esheep.com/apphttps://openart.ai/workflows/all

单链表OJ题(课堂总结)

1.链表的带环问题 上图就是一个典型的带环链表 1.1如何判读链表是否带环&#xff1f; 最常见的方法就是利用快慢指针&#xff0c;快指针追加慢指针&#xff0c;当二者相等的时候即可判断链表带环 其实现的代码如下&#xff1a; bool hasCycle(struct ListNode*head) { s…

万博智云×华为云 | HyperBDR云容灾上架,开启联营联运新篇章

日前&#xff0c;万博智云HyperBDR云容灾正式入驻华为云云商店&#xff0c;成为华为云基础软件领域联营联运合作伙伴。通过联营联运&#xff0c;双方将进一步加深在产品、解决方案、渠道拓展等多方面的强强联合&#xff0c;为企业提供更加安全、高效的数据保护解决方案&#xf…

全面解析Java.lang.ClassCastException异常

全面解析Java.lang.ClassCastException异常 全面解析Java.lang.ClassCastException异常&#xff1a;解决方案与最佳实践 &#x1f680;&#x1f4da;摘要引言1. 什么是Java.lang.ClassCastException&#xff1f;代码示例 2. 报错原因2.1 类型不兼容2.2 泛型类型擦除2.3 接口和实…

HTTP -- HTTP概述

HTTP概述 HTTP使用的是可靠的数据传输协议。 web内容都是存储在web服务器上的&#xff0c;web服务器所使用的是http协议&#xff0c;故被称为http服务器。 web服务器是web资源的宿主&#xff0c;web资源是web内容的源头。 因特网上有数以千种的数据类型&#xff0c;http仔细的…

文心智能体之情感领航员:你的智能情感导师

文章目录 引言情感领航员的诞生与定位情感领航员的优势与特点专业性个性化便捷性隐私保护 如何创建自己的智能体创建方式智能体名称和设定基础配置角色与目标指导原则限制澄清个性化 高级配置保存 /发布流量数据分析智能体调优 总结 引言 在现代社会中&#xff0c;情感问题的普…

Docker拉取镜像报错:x509: certificate has expired or is not yet v..

太久没有使用docker进行镜像拉取&#xff0c;今天使用docker-compose拉取mongo发现报错&#xff08;如下图&#xff09;&#xff1a; 报错信息翻译&#xff1a;证书已过期或尚未有效。 解决办法&#xff1a; 1.一般都是证书问题或者系统时间问题导致&#xff0c;可以先执行 da…