数据结构二叉树顺序存储——堆

  • 1.堆的概念
  • 2.堆的实现 (建小堆为例)
    • 2.1 初始化和销毁
    • 2.2 判空
    • 2.3 获得堆顶元素和堆的大小
    • 2.4 插入
    • 2.5 删除
  • 3.堆的构建(建小堆为例)

1.堆的概念

将若干数据或元素按照完全二叉树的存储方式顺序存储到一个一维数组中,在逻辑结构中(完全二叉树)满足每个根结点的值不大于或不小于它的左右孩子结点的值。满足上述条件称为小堆或大堆。

堆的性质:堆中某节点的值不大于不小于双亲结点的值;堆在逻辑结构上是一棵完全二叉树。

比如下图是一个小堆 (数据的增删都以这个数组为例)
在这里插入图片描述

堆只需要满足双亲结点的值不大于或不小于孩子结点的值即可,左右孩子结点的值不做要求。比如上图的13和17互换位置依然是小堆。

注意,如果元素在数组中有序,则一定可以构成堆,反之不成立。

2.堆的实现 (建小堆为例)

由于堆的物理结构使用的是数组,因此定义的结构体和顺序表类似

typedef int heapDataType;typedef struct heapNode
{heapDataType* array;int size;int capacity;
}heapNode;

关于堆的接口,包括初始化和销毁,插入,删除,判空,以及获取堆顶元素和堆的大小。

2.1 初始化和销毁

和顺序表非常类似,就不赘述了,重点放在插入数据和删除数据。

//初始化堆
void heapInit(heapNode* hp)
{assert(hp);hp->array = NULL;hp->capacity = hp->size = 0;
}//销毁堆
void heapDestroy(heapNode* hp)
{assert(hp);hp->array = NULL;hp->capacity = hp->size = 0;
}

2.2 判空

由于结构体内有成员变量size,直接返回size是否等于0即可

//判空
bool heapIsEmpty(heapNode* hp)
{assert(hp);return hp->size == 0;
}

2.3 获得堆顶元素和堆的大小

根就是堆顶,在物理结构上,堆顶就是数组的首元素。因此返回数组首元素即可。

//获得堆顶数据
heapDataType heapTop(heapNode* hp)
{assert(hp);//堆不能为空assert(!heapIsEmpty(hp));return hp->array[0];
}//获得堆的大小
int heapSize(heapNode* hp)
{assert(hp);return hp->size;
}

2.4 插入

增加数据的时候,增加在数组的最后一个位置,增加数据前为堆,增加数据后也需要还是堆。
但数据的不同,可能影响结构。
在这里插入图片描述
在数组末尾增加一个大于或等于17的值时,堆的结构不会被破坏,如果增加的数据小于17,则需要将其调整为堆。

调整需要用到向上调整算法。指的是将符合条件的数据进行上移。
调整过程为,和该结点的双亲结点的值进行比较,如果小于双亲结点的值,那么进行交换,再作为新的孩子和新的双亲结点的值比对,直到换到根节点为止。

以插入一个7为例,调整过程如下图所示。
当调整的结点处于根节点时,则不需要进行下去了,即child下标为0
在这里插入图片描述
调整过程中可能涉及到的结点为从根节点到插入的新结点上的一条路径上的结点。
我们知道,在完全二叉树中,任一结点(设下标为k)的双亲结点的下标为(k-1)/ 2

代码实现如下

//交换
void swap(heapDataType* x1, heapDataType* x2)
{heapDataType tmp = *x1;*x1 = *x2;*x2 = tmp;
}//向上调整算法
void AdJustUp(heapDataType* array,int child)
{//双亲节点的下标int parent = (child - 1) / 2;while (child > 0){if (array[child] < array[parent]){//交换数据swap(&array[child], &array[parent]);//更新下标child = parent;parent = (child - 1) / 2;}else{break;}}
}//插入
void heapPush(heapNode* hp, heapDataType x)
{assert(hp);//容量不够,扩容if (hp->size == hp->capacity){int newCapacity = hp->capacity == 0 ? 4 : hp->capacity * 2;//空间可能开辟失败,为保留原数据,创建新的指针变量heapDataType* tmp = realloc(hp->array, sizeof(heapNode) * newCapacity);if (tmp == NULL){return;}//开辟成功hp->capacity = newCapacity;hp->array = tmp;}//插入数据hp->array[hp->size++] = x;//向上调整重新建堆AdJustUp(hp->array,hp->size-1);
}

插入数据和顺序表中完全一样,只是多了一步向上调整。

2.5 删除

堆的删除指的是删除堆顶的数据。对于数组,我们比较容易想到的是数据的挪动覆盖,但是这样会导致一个问题,就是删除数据后,堆结构被破坏的很严重。使得难以将其恢复成堆。因此这种做法不合适。

这里的做法是,将堆顶数据和最后一个数据进行交换,然后在删除。这样做虽然也会破坏堆的结构,但是破坏的程度没有那么大,可以通过算法(向下调整算法)将其恢复成堆。

刚刚我们插入了一个7,现在我们在删除一次堆顶的数据。

首先,交换首尾数据并删除,从根节点开始向下调整,找到左右孩子结点中值小的那个进行交换,在作为新的双亲结点,找新的值较小的孩子结点进行交换,直到换到叶子结点。

举例删除7,删除过程如图所示。
在这里插入图片描述
如果25处数据是15,则在进行一次交换即可。
有几个小问题需要处理,1.怎么找到较小的孩子结点。2.调整的结束条件。

对于问题1:可以使用假设法,假设左孩子值较小,在和右孩子的值进行比较,如果存在右孩子且右孩子值小,则下标加一
对于问题2:在数组中,孩子结点的下标不会超过数组大小。如果孩子结点的下标已经超过数组,说明调整的结点已经是叶子结点,则不要在进行调整了。

代码如下

//向下调整
void AdJustDown(heapDataType* array, int num_array, int parent)
{//孩子下标int child = parent * 2 + 1;while (child < num_array){//找到值小的孩子,if条件1指存在右孩子,条件2指右孩子值小if (child + 1 < num_array && array[child + 1] < array[child]){++child;}if (array[child] < array[parent]){//交换数据swap(&array[child], &array[parent]);//更新下标parent = child;child = parent * 2 + 1;}else{break;}}
}//删除
void heapPop(heapNode* hp)
{assert(hp);//堆不能为空assert(!heapIsEmpty(hp));//交换堆的首尾数据swap(&hp->array[0], &hp->array[hp->size - 1]);hp->size--;//向下调整重新建堆AdJustDown(hp->array, hp->size, 0);
}

3.堆的构建(建小堆为例)

随机给一个数组,将这个数组构建成为堆。
这里有两种方法,第一种是依次插入数据,向上调整建堆。一个数据可以看作堆,依次插入数据后,堆结构被破坏,进行向上调整,数组遍历完全后,就构建成了堆。

int main()
{srand(time(NULL));heapNode hp;heapInit(&hp);//生成十个随机数,插入堆for (int i = 0; i < 10; i++){int j = rand() % 100;heapPush(&hp, j);}//打印,数据符合小堆for (int i = 0; i < 10; i++){printf("%d ", hp.array[i]);}printf("\n\n");//该操作为获取堆顶数据,打印后,在删除,最终结果为升序while (!heapIsEmpty(&hp)){printf("%d ", heapTop(&hp));heapPop(&hp);}heapDestroy(&hp);return 0;
}

打印结果如图
在这里插入图片描述

第二种方法为向下调整建堆。
向下调整算法有一个前提,指的是调整的结点的左右子树需要是堆。因此从根节点开始调整的方法不适用了,一个值可以看成是堆,因此从倒数的第一个非叶子结点开始向下调整。
在这里插入图片描述
将该数组打乱成25,36,10,13,17,42

向下调整建堆的过程如下图所示
在这里插入图片描述

那么第一个非叶子结点下标是多少呢?
首先,最后一个非叶子结点是最后一个结点的双亲结点,下标需要结合数组和完全二叉树的性质来计算。假设数组中元素个数为n,则最后一个元素下标为n-1,完全二叉树中,下标为n-1的双亲结点的下标为[ ( n - 1 ) -1 ] / 2 即 ( n - 2)/ 2

调整后数组就符合堆结构了。

int main()
{int arr[10] = { 0 };//随机生成二十个树for (int i = 0; i < 10; i++){arr[i] = rand() % 100 + 10;}//打印随机数for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}//向下调整建堆for (int j = (10 - 2) / 2; j >= 0; --j){AdJustDown(arr, 10, j);}printf("\n\n");//打印调整后的数组for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}
}

打印结果如图
在这里插入图片描述

堆的介绍就先到这里了。
本篇主要介绍二叉树的顺序存储,也就是堆了,关于堆的应用(堆排序,topk问题)就留到下次在介绍吧。

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

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

相关文章

深度学习armv8/armv9 cache的原理

快速链接: 【精选】ARMv8/ARMv9架构入门到精通-[目录] &#x1f448;&#x1f448;&#x1f448; 本文转自 周贺贺&#xff0c;baron&#xff0c;代码改变世界ctw&#xff0c;Arm精选&#xff0c; 资深安全架构专家&#xff0c;11年手机安全/SOC底层安全开发经验。擅长trustzon…

【pytest、playwright】多账号同时操作

目录 方案实现思路&#xff1a; 方案一&#xff1a; 方案二&#xff1a; 方案实现思路&#xff1a; 依照上图所见&#xff0c;就知道&#xff0c;一个账号是pytest-playwright默认的环境&#xff0c;一个是 账号登录的环境 方案一&#xff1a; 直接上代码&#xff1a; imp…

Node.js-------初识Node.js与内置模块

能够知道什么是 Node.js能够知道 Node.js 可以做什么能够说出 Node.js 中的 JavaScript 的组成部分能够使用 fs 模块读写操作文件能够使用 path 模块处理路径能够使用 http 模块写一个基本的 web 服务器 一.初识Node.js 1.浏览器中的 JavaScript 的组成部分 2.Node.js 简介 …

使用Pollard_rho算法分解质因数

分解质因数的朴素算法 最简单的算法即为从 [2, sqrt&#xff08;N&#xff09;] 进行遍历。 vector<int> breakdown(int N) {vector<int> result;for (int i 2; i * i < N; i) {if (N % i 0) { // 如果 i 能够整除 N&#xff0c;说明 i 为 N 的一个质因子。…

iOS苹果签名共享签名是什么以及如何获取?

哈喽&#xff0c;大家好呀&#xff0c;咕噜淼淼又来和大家见面啦&#xff0c;最近有很多朋友都来向我咨询共享签名iOS苹果IPA共享签名是什么&#xff0c;针对这个问题&#xff0c;淼淼来解答一下大家的疑惑并告诉大家iOS苹果ipa共享签名需要如何获取。 现在苹果签名在市场上的…

Autodesk Maya 2025---智能建模与动画创新,重塑创意工作流程

Autodesk Maya 2025是一款顶尖的三维动画软件&#xff0c;广泛应用于影视广告、角色动画、电影特技等领域。新版本在功能上进行了全面升级&#xff0c;新增了对Apple芯片的支持&#xff0c;建模、绑定和角色动画等方面的功能也更加出色。 在功能特色方面&#xff0c;Maya 2025…

CVPR 2024 | 从6篇论文看扩散模型diffusion的改进方向

1、Accelerating Diffusion Sampling with Optimized Time Steps 扩散概率模型&#xff08;DPMs&#xff09;在高分辨率图像生成方面显示出显著性能&#xff0c;但由于通常需要大量采样步骤&#xff0c;其采样效率仍有待提高。高阶ODE求解在DPMs中的应用的最新进展使得能够以更…

从PDF到高清图片:一步步学习如何转换PDF文件为高清图片

引言 PDF文件是一种便携式文档格式&#xff08;Portable Document Format&#xff09;&#xff0c;最初由Adobe Systems开发&#xff0c;用于在不同操作系统和软件之间保持文档格式的一致性。PDF文件通常包含文本、图片、图形等多种元素&#xff0c;并且可以以高度压缩的方式存…

Redis配置与优化

目录 引言 一、关系型数据库与非关系型数据库 1、关系型数据库 2、非关系型数据库 3、关系型数据库和非关系型数据库的区别 1.数据存储方式不同 2.扩展方式不同 3.对事物性的支持不同 4、非关系型数据库产生背景 二、Redis简介 1、Redis优点 2、Redis为什么这麽快&…

hcip实验4:gre mgre ppp综合实验

实验拓扑: 实验目的&#xff1a; 1.R5为ISP&#xff0c;只能进行IP地址配置&#xff0c;其所有地址均配为公有IP地址 2.R1和R5间使用PPP的PAP认证&#xff0c;R5为主认证方&#xff1b;R2与R5之间使用ppp的CHAP认证&#xff0c;R5为主认证方;R3与R5之间使用HDLC封装; 3.R1、R…

vue+elementUI搭建动态表头的表格

前提&#xff1a;以下代码是vue2项目结合elementUi完成的 数据结构 后端传来的数据是两个list&#xff0c;一个表头的list&#xff0c;一个表格内容的list // 表头 headTableAtts: [{ columnLabel: 姓名, columnName: name },{ columnLabel: 年龄, columnName: age },{ colu…

计算机网络面试问题(一)

1.在浏览器中输⼊URL并按下回⻋之后会发⽣什么 2.TCP三次握⼿的过程,为什么三次握手 TCP&#xff08;传输控制协议&#xff09;的三次握⼿是建⽴⽹络连接的过程&#xff0c;确保通信双⽅能够正确地进⾏数据传输。 第⼀次握⼿&#xff08;SYN&#xff09;&#xff1a; 客户端&am…

[羊城杯 2020]EasySer

[羊城杯 2020]EasySer 进入页面&#xff0c;发现是ubuntuapache2&#xff0c;但是好像没啥用 尝试访问/robots.txt&#xff0c;得到 访问/star1.php/&#xff0c;查看源码&#xff0c;得到提示 一看就知道是ssrf&#xff0c;使用http://127.0.0.1/ser.php&#xff0c;得到…

Spring日志框架

前言 本文我们简单说说关于Spring中的日志框架,以及对应的注解 我们知道,公司服务器在运行的时候,一定会打印日志,有很多优点,比如预防报警,或者是某重大事故尝试修复等等都需要查看日志 应该说日志对我们来说并不陌生,我们在之前刷题或者是程序遇到bug的时候也经常会将程序的状…

windows 系统图标 桌面刷新 位置变化解决办法

Windows操作系统下&#xff0c;系统图标由于是内置图标&#xff0c;即使桌面关闭了图标自动排列&#xff0c;在桌面右键刷新或系统重启后&#xff0c;依然会位置自动改变&#xff0c;有时候确实需要管理图标&#xff0c;这种自动变化就特别烦&#xff0c;怎么办呢&#xff1f; …

uniapp微信小程序消息订阅详解

一、微信公众平台申请订阅模板 注意&#xff1a;订阅信息 这个事件 是 当用户 点击的时候触发 或者 是 支付成功后触发&#xff0c; 用户勾选 “总是保持以上选择&#xff0c;不再询问” 之后或长期订阅&#xff0c;下次订阅调用 wx.requestSubscribeMessage 不会弹窗&#xf…

如何选择家用洗地机?四大性能出色产品,新手必看

在现代生活中&#xff0c;随着人们对健康和卫生意识的提高&#xff0c;家庭清洁变得越来越重要。然而&#xff0c;传统的清洁方式往往效率低下&#xff0c;难以满足需求。幸运的是&#xff0c;现代科技的发展为我们带来了许多智能清洁设备&#xff0c;其中洗地机就是一种非常实…

Qt 富文本处理 (字体颜色大小加粗等)

Qt中支持HTML的控件有textEdit 、label 、textBrowser 。 接口&#xff1a;setHtml("Qt"); toHtml(). 文本样式设置 : 可分字设置 &#xff0c;主要使用QTextCharFormat类进行文本样式设置。 示例&#xff1a; QTextCharFormat fmt; //粗体 fmt.setFontWeight…

数据结构:归并排序

归并排序 时间复杂度O(N*logN) 如果两个序列有序,通过归并,可以让两个序列合并后也有序,变成一个有序的新数组 对于一个数组,如果他的左右区间都有序,就可以进行归并了 归并的方法 将数组的左右两个有序区间比较,每次都取出一个最小的,然后放入临时数组(不能在原数组上修改…

《自动机理论、语言和计算导论》阅读笔记:p68-p114

《自动机理论、语言和计算导论》学习第4天&#xff0c;p68-p114总结&#xff0c;总计47页。 一、技术总结 1.inverted indexes 明白单词的意思是“反转的索引”&#xff0c;但是不明白其在书中具体指什么&#xff0c;去查询资料的话需要花很不多时间&#xff0c;先继续往下看…