C数据结构-堆的实现思路和堆排序的实现

news2024/9/25 1:25:46

堆和堆排

  • 堆的定义
    • 为什么使用数组?
    • 堆接口函数的实现
      • 堆的初始化
      • 堆的销毁
      • 堆的打印
      • 堆的插入!!
      • 堆的删除!!
      • 堆的判空
      • 返回堆顶的元素
      • 堆的大小
  • 堆排序的实现!!
    • 实现堆排序的两种方式
    • 时间复杂度的分析
  • Last

前言:
在了解完二叉树的相关概念之后,我们就可以来学习下一个数据结构的实现了—堆!
源代码放在最后,需要的自取

堆和前面的数据结构比起来,对小白来说还是有一些难度的,因为堆的逻辑结构看起来更适合用链表来实现,但是实际却是使用数组更加的好一点(后面分析),这就导致后面的内容可能不是那么的好理解,当然比较复杂的部分,我都会结合画图和动图的方式,给大家展示出来!不必担心。

堆的定义

堆(Heap)的定义:
现实中我们通常把堆(一种二叉树)使用顺序结构的数组来存储。
堆总是满足下列性质:

  1. 堆中某个结点的值总是不大于或不小于其父结点的值
    堆的某个结点总是比其父亲结点小的,称为大堆,如下图:
    在这里插入图片描述

反之被称为小堆,如下图:
在这里插入图片描述

  1. 堆总是一棵完全二叉树
    这也是堆能使用数组实现的一大重要的特征。

了解过二叉树的同志们,肯定知道,对于普通的二叉树肯定更加的适合用链式的存储方式来链接,但是,对于完全二叉树,使用数组会更加的高效一些(当然也可以是同链式存储)。

为什么使用数组?

因为数组在内存中是连续开辟空间的,所以可以更好的随意读取想要的元素,而且能更好的找出最大的前几个数,链式结构的话,就必须每次都遍历一边链表。假如要在某个位置插入或删除的元素的话,链式结构还需要用一个指针来记录遍历指针的前一个位置,以达到删除和添加元素的目的。

比起链式的存储结构,使用数组也会让我们在后面实现堆的各个函数接口上,简单很多(比如返回堆低的元素)。

下面我们就来谈谈怎么把一个完全二叉树(堆)的数据放入数组中。

从上到下,从左到右
将堆顶的第一个元素作为数组的第一个元素存放在数组中去,然后是第二层从左到右依次存放到数组,然后到第三层…依次类推。
图解如下:
在这里插入图片描述

同样要从数组中将数据取出来,画成逻辑图的话,也是这种方式,所以在学习堆的时候,看不懂的地方要多画图出来。同时大家看到一个数组的时候,也要能看到上图右边的逻辑结构图。

堆接口函数的实现

这里先将所有的接口函数给大家展示出来,让大家大概的明白后面我们要实现那些功能。另外需要解释的一点就是堆的初始化代码。

因为堆使用数组实现,所以使用一个指针来指向存放数据的数组,因为要考虑到扩容的问题,所以并没用选择直接使用一个数组来存放数据。同样的使用一个szie变量来表示当前堆中的数据个数,使用capacity表示数组容量的大小。

一下就是Heap.h的所有内容
在这里插入图片描述

堆的初始化

接下来就是各个函数接口的实现问题了,我们先来看第一个,堆的初始化问题。
堆的初始化其实很简单,根我们前面所实现的动态顺序表很相似。
在这里插入图片描述
我们这里给定的数组的初始大小为4个int。

堆的销毁

因为是使用的数组,不需要像链表那样一个一个释放。销毁堆的时候直接free掉就好了。
在这里插入图片描述

堆的打印

因为size是表示数组中的数据个数,同时也是最后一个数据位置的下一个位置,比如数组此时有两个数据,那么size==2,就如下图:
在这里插入图片描述
所以我们通过控制size就可以打印出,堆中的数据。
在这里插入图片描述

堆的插入!!

堆的插入和删除是实现堆的重点内容,因为我们要保证的是,在插入一个数据和删除一个数据,之后剩下的数据,仍然是一个大堆或小堆(我们依实现大堆来写代码)。

这个部分的讲解要结合堆的逻辑图来看,假如我们要在下图中的堆中插入一个新的数据 55 。
在这里插入图片描述
新插入的数据肯定是插入到数组的最后面(假设数组已经扩容完成了),那么55的位置应该如下图:
在这里插入图片描述
此时再去看这个堆,已经不符合堆的定义了,堆要么是大堆(所有的结点都比父亲结点要小)要么是小堆(所有的结点都比父亲结点要大)。而55的到来,明显的破坏了原来大堆的结构,怎么办呢?

首先,我们将55和它的父亲结点进行比较,如果比起父亲结点要大的话,就交换彼此然后再和新的父亲结点进行比较…知道登顶或者没有其父亲结点大,就停止。
图解如下:
在这里插入图片描述
如此以来就顺利的完成了堆的插入。

那么大思路确定了之后,我们就来考虑怎么去找到某个结点的父亲结点。观察上图中的红色数字我们可以发现一个规律(父亲结点我们用paent表示,当前结点,即孩子结点用child表示)

堆中数据的孩子和父亲结点的关系满足以下表达式:
leftchild = 2parent+1;
rightchild = 2
parent+2;
parent = (child-1) / 2;

在这里插入图片描述
不必关心该公式怎么来的。

有了这个重要的条件之后我们就能来着手写代码了。

同样的在插入数据之前要先检查扩容的问题
在这里插入图片描述
我们重点来看,Adjuaiup函数的实现。该函数的前半部分看不懂的建议重新去复习一下该专栏的顺序表的实现。

刚才我们已经将整体大的思路讲了,但是对于循环比较父亲结点的结束条件还没说,下面我们还是以55为例来看一下完成的插入过程。

在这里插入图片描述
我们可以看到,当child<0的时候,代表已经将需要比较的父亲结点全部比较完了,循环也就结束了。
下面给出代码,对于交换的部分我们还是封装成一个函数,因为后面还需要用到交换的功能。
在这里插入图片描述
在这里插入图片描述
在实现交换的函数的时候,注意传的是地址,因为形参的改变并不影响实参!!!

到此堆的插入就结束了,下面是测试代码和结果:
在这里插入图片描述
大家可以自己将测试的用例的逻辑结构画出来看一下是不是大堆。

堆的删除!!

如果插入完全理解了之后,对于删除其实也大差不差。

我们要删除的是堆顶的数据,删除完成之后保证剩下的数据还是大堆的结构。以上面的测试用例数据为例,我们来看一下如何完成删除。

删除的思路我们就不拐弯抹角了,直接给出

  1. 堆顶和堆低的元素交换,size–,删除堆低(原堆顶)的数据。
  2. 通过根结点找到两个孩子结点,
  3. 选出两个孩子结点较大的那个和根节点进行比较
  4. 交换
  5. 更新父亲结点更新孩子结点
  6. …依次类推

图解如下:
在这里插入图片描述
关于怎么通过孩子找父亲和通过父亲找孩子,在上面我们已经说了。
接下来就是调整函数的实现了。

相信这个过程大家不难理解,接下来就是代码,大家可以试着跟着代码走一遍以加深理解。
在这里插入图片描述
删除数据时候,需要注意的一点就是,如果堆中没有数据,就不能删除,所以要加上一个assert(php->size > 0)断言一下,确保删除是有效的。

以上就是堆的删除。

堆的判空

解决了堆的两大难题之后,后面的都是虾兵蟹将了,一看就懂,直接上代码。
在这里插入图片描述

返回堆顶的元素

在这里插入图片描述

堆的大小

在这里插入图片描述

堆排序的实现!!

学完堆的实现之后,我们就来看一中与有关的排序–堆排序。

通过前面的学习我们了解到,给定如何一个数组,我们都可以将它看作是堆这种数据结构。由此衍生出了,堆排序。

堆排要做的事情就是将一组数组变成大堆或者小堆。

实现堆排序的两种方式

前面我们已经实现了两种调整堆的方式,我们先来使用第一种Adjustup(向上调整)来看一下怎么将凌乱的数据调整成堆的。
向上调整的方式是通过孩子结点去找父亲结点。
所以在每次调用Adjustup函数的时候,都需要将i作为孩子结点传入。
在这里插入图片描述
这个过程也就是将数组中的数据一个个插入的方式建堆。其时间复杂度我们等会再分析。

另外一种方式就是用Adjustdown来实现堆。
Adjustdown建堆的方式就不一样了,我们用动图来展示Adjustdown的建堆过程。

  1. 先通过最后一个元素找到其父亲结点
  2. 比较大小后交换,然后找倒数第二个数据
  3. 找到其父亲结点,再比较进行交换。
  4. 找到倒数第三个结点…依次类推。

图解:
在这里插入图片描述
下面是代码实现。
在这里插入图片描述
sz是数组大小,sz-1是找到最后一个元素,再-1, / 2 是找到父亲结点。
通过控制i–,将数组中的所有数据与其父亲结点进行比较依次来达到建堆的目的。

下面是这两种方式的完整代码。
在这里插入图片描述
下面就是时间和空间复杂度的分析。

时间复杂度的分析

判断一个算法的好坏我们总是要通过时间复杂度来分析的。

我们先来看第一种的方式的时间复杂度。
第一种方式有两个过程,一是通过控制 i-- 是使每个数据都向上调整,第二就是数据的调整过程了。
第一个过程不难理解,就是O(n),那么我们依最坏的请况来看第二,向上调整过程的时间复杂度。

我们之前谈过完全二叉树的高度和结点问题,得出一个表达式
在这里插入图片描述
n表示结点的个数,h表示二叉树的高度。

谈论时间复杂度都是以最坏的情况讨论,所以我们就暂时将完全二叉树看作满二叉树。
那么得出:
在这里插入图片描述
所以一个数据最坏的调整时间为logn,那么最坏的情况就是每个数据都需要向上调到堆低才能结束,所以第一种堆排的时间复杂度就是O(n*logn)

第二种方式就简单多了,因为我们是遍历每一个数据,将其与父亲结点进行比较,如果大于父亲结点就交换,否则就i–,找到下一个数据再与其父亲结点进行交换。

因此第二种方式的时间复杂度为O(n)

Last

堆的实现代码:
堆的实现
堆和顺序表以及链表比起来。是比较抽象的,前面的数据结构的逻辑图和实际的物理存储方式差不多,看着图就能明白大概的实现过程,但是由于堆的逻辑图和实际选择的存储方式差别太大,所以显得比较的抽象。不过真正理解了过后其实也就习惯了,学习就是逐渐的掌握技能的过程,希望你最后的结果,对得起一路的坎坷。

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

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

相关文章

【STC8A8K64D4开发板】——STC8A8K64D4开发板介绍

版权声明&#xff1a;本文为博主原创文章&#xff0c;转载请附上原文出处链接。 文章目录前言一、STC8A8K64D4系列单片机介绍二、STC8A8K64D4开发板概述三、STC8A8K64D4开发板硬件框图四、STC8芯片命名规则五、STC8A8K64D4核心板-2款对比六、STC8A8K64D4开发板-功能描述七、STC…

肠道微生物群与过敏性鼻炎

过敏性鼻炎是个全球性健康问题&#xff0c;易反复发作&#xff0c;困扰着全世界大约着4亿人。自20世纪60年代以来&#xff0c;过敏性鼻炎的患病率持续上升。近几年&#xff0c;我国主要大中城市过敏性鼻炎的平均患病率从11.1%升高到17.6%。 过敏性鼻炎主要临床表现包括发痒、打…

相比Superset和Metabase,DataEase开源工具为什么更易用?

企业业务环境或应用环境中产生的数据大多存储在数据库中&#xff0c;但是这些数据无法直接、形象地加以展示。数据可视化的意义就在于将原始数据通过不同类型的图形进行展示&#xff0c;为数据分析提供更加具象化的手段&#xff0c;这是数据可视化分析工具诞生的使命和意义。 …

[足式机器人]Part3机构运动微分几何学分析与综合Ch02-1 平面机构离散运动鞍点综合——【读书笔记】

本文仅供学习使用 本文参考&#xff1a; 《机构运动微分几何学分析与综合》-王德伦、汪伟 《微分几何》吴大任 Ch02-1 平面机构离散运动鞍点综合2 平面运动微分几何学2.1 平面离散运动的矩阵表示2.2 鞍点规划2 平面运动微分几何学 平面连杆机构由连架杆、机架和连杆组成&#…

运筹说 第73期 | 图论创始人“数学之王”一 欧拉

前面我们介绍了有关动态规划的相关内容&#xff0c;相信大家也都有了一些收获&#xff0c;下面我们学习的列车继续驶往“图与网络分析”的站点&#xff0c;在本次文章中我们将一起走近图论的奠基人——欧拉Leonhard Euler&#xff0c;希望能给大家学习运筹学的旅程中带来不一样…

甘露糖-聚乙二醇-6-羧甲基荧光素mannose-PEG-6-FAM

甘露糖-聚乙二醇-6-羧甲基荧光素mannose-PEG-6-FAM 中文名称&#xff1a;甘露糖-6-羧甲基荧光素 英文名称&#xff1a;mannose-6-FAM 别称&#xff1a;6-羧甲基荧光素标记甘露糖&#xff0c;6-羧甲基荧光素-甘露糖 PEG分子量可选&#xff1a;350、550、750、1k、2k、34k、5…

互联网基础结构发展的三个阶段及其特点

互联网的基础结构大体上经历了三个阶段的演进。但这三个阶段在时间划分上并非截然分开而是有部分重叠的&#xff0c;这是因为网络的演进是逐渐的而不是在某个日期突然发生了变化。 第一个阶段是从单个网络ARPANET向互连网方向发展。这一阶段的主要特点是TCP/IP协议初步成型。1…

Apollo 应用与源码分析:CyberRT-protobuf

目录 概念 特点 优点 缺点 文件的创建 1.字段规则 2.数据类型 3.字段名称 4.字段编号 文件的编译 protobuf 编译命令编译 protobuf cmake 方式编译 使用bazel 编译 在protobuf 文件夹下创建build 文件 代码解释&#xff1a; 样例 protobuf 使用文件 代码解释…

Python:入门与基本语法

目录 一、Python环境 官方下载 Python开发工具 插件安装 二、基本数据类型 动态语言的体现 静态语言的体现 弱语言的体现 强语言的体现 三、基本数据类型 java八大基本数据类型 Python四大基本数据类型 案例 基本数据类型 test01 源码 引用数据类型 列表 test02 源码…

你不知道的SQL语言数据库原理

1、SQL的概述 SQL全称&#xff1a; Structured Query Language&#xff0c;是结构化查询语言&#xff0c;用于访问和处理数据库的标准的计算机语言。 SQL语言1974年由Boyce和Chamberlin提出&#xff0c;并首先在IBM公司研制的关系数据库系统SystemR上实现。 美国国家标准局(AN…

【配电网重构】负荷平衡的配电网重构【含Matlab源码 2180期】

⛄一、 负荷平衡的配电网重构 1 引言 配电网网络重构是指在正常或非正常运行条件下通过改变开关的开关状态来改变网络的拓扑结构。配电网故障恢复供电是指配电网发生故障后,在故障已被定位和隔离的基础上,研究如何恢复对无故障停电区域供电的问题,本文简称为故障恢复。故障恢复…

Spring Cloud Alibaba 整合 Nacos

写在最前 项目 GitHub 地址 mingyue-springcloud-learning 【mingyue-springcloud-user、mingyue-springcloud-member】 阅读推荐 Spring Cloud 入门必读Spring Cloud Alibaba 入门必读 版本声明 spring-boot: 2.7.5 spring-cloud: 2021.0.5 spring-cloud-alibaba: 2021.0.4…

论文阅读 Fast Reinforcement Learning Via Slow Reinforcement Learning

论文阅读 RL^2 Fast Reinforcement Learning Via Slow Reinforcement Learning1. 摘要2.introduction3. 实现4.小结1. 摘要 强化学习可以对于单个任务有较好的效果&#xff0c;但需要大量的尝试。动物往往可以通过少量的尝试就获得很好的效果。原因在于动物可以更好地使用先验…

学习笔记——Java Stream 源码学习

思路 先上一段代码 List<User> userList new ArrayList<>();for (int i 0; i < 10; i) {userList.add(new User(i, "wtq", "1234"));}userList.stream().filter(user -> user.getUserId() > 3).filter(user -> user.getUserId(…

【JVM】字节码技术:图解字节码形式下的 方法执行流程

一、源文件 package cn.itcast.jvm.t3.bytecode;/** * 演示 字节码指令 和 操作数栈、常量池的关系 */ public class Demo3_1 {public static void main(String[] args) {int a 10;int b Short.MAX_VALUE 1;int c a b;System.out.println(c);} }二、反编译的字节码文件 …

《Go语言精进之路,从新手到高手的编程思想、方法和技巧1》读书笔记和分享

Go语言精进之路&#xff0c;从新手到高手的编程思想、方法和技巧读书分享1 本书定位2 本书内容总览3 选择本书的原因4 小收获分享第7-12条 真的不知道咋命名第13-19条 能用——怎么用更好5 个人总结第一部分 熟知 Go 语言的一切第 1 条 了解 Go 语言的诞生与演进第 2 条 选择适…

甘露糖-聚乙二醇-CY5.5 /Cy5.5-PEG-mannose

甘露糖-聚乙二醇-CY5.5 /Cy5.5-PEG-mannose 中文名称&#xff1a;甘露糖-近红外染料CY5.5 英文名称&#xff1a;mannose-Cyanine5.5 别称&#xff1a;CY5.5修饰甘露糖&#xff0c;CY5.5-甘露糖 PEG分子量可选&#xff1a;350、550、750、1k、2k、34k、5k 包装&#xff1a;…

es(网站的搜索技术)

一。搜索技术 springboot集成es ElasticsearchRestTemplate mapping 继承 extends ElasticsearchRepository 实体类写的映射关系&#xff1a; 注解&#xff1a;Document用来声明Java对象与ElasticSearch索引的关系 indexName 索引名称(是字母的话必须是小…

线性回归的神经网络法——机器学习

一、算法思想 线性回归方程在神经网络深度学习中线性回归方程是需要掌握的最基础的式子&#xff0c;就是&#xff1a;ywxb,其中w,b是未知的。 神经网络就是可以通过收集大量的数据集&#xff0c;然后将这些数据集进行训练后得到几个较为准确的参数&#xff0c;训练数据集后会得…

MongoDB单机集群方案及详解

目录帮助文档MongoDB在企业级网站中的定位单机MongoDB部署集群&#xff08;副本集&#xff09;集群&#xff08;副本集仲裁&#xff09;集群&#xff08;分片&#xff09;mongo运行原理mongo管理小工具推荐帮助文档 MongoDB官方更新速度过快&#xff0c;语法不断更新&#xff…