当前位置: 首页 > news >正文

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

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

        本篇内容适合已经学完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为调堆最后一个元素}
}

相关文章:

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

前言:堆算是一种相对简单的数据结构&#xff0c; 本篇文章将详细的讲解堆中的知识点&#xff0c; 包括那些我们第一次学习堆的时候容易忽略的内容&#xff0c; 本篇文章会作为重点详细提到。 本篇内容适合已经学完C语言数组和函数部分的友友们观看。 目录 什么是堆 建堆算法…...

JS数据类型运算符标准库

目录 数据类型运算符标准库对象Object对象属性描述对象Array对象包装对象Boolean对象Number对象String对象Math对象Date对象...

单片机之从C语言基础到专家编程 - 4 C语言基础 - 4.13数组

C语言中&#xff0c;有一类数据结构&#xff0c;它可以存储一组相同类型的元素&#xff0c;并且可以通过索引访问这些元素&#xff0c;没错&#xff0c;这类数据结构就是数组。数组可以说是C语言中非常重要的数据结构之一了。使用数组可以是程序逻辑更加清晰&#xff0c;也更加…...

【码银送书第二十期】《游戏运营与出海实战:策略、方法与技巧》

市面上的游戏品种繁杂&#xff0c;琳琅满目&#xff0c;它们是如何在历史的长河中逐步演变成今天的模式的呢&#xff1f;接下来&#xff0c;我们先回顾游戏的发展史&#xff0c;然后按照时间轴来叙述游戏运营的兴起。 作者&#xff1a;艾小米 本文经机械工业出版社授权转载&a…...

String 类

目录&#xff1a; 一. 认识 String 类 二. String 类的基本用法 三. String对象的比较 四.字符串的不可变性 五. 认识 StringBuffer 和 StringBuilder 一. 认识 String 类&#xff1a; 在C语言中已经涉及到字符串了&#xff0c;但是在C语言中要表示字符串只能使用字符数组或者…...

Chromebook Plus中添加了Gemini?

Chromebook Plus中添加了Gemini&#xff1f; 前言 就在5月29日&#xff0c;谷歌宣布了一项重大更新&#xff0c;将其Gemini人工智能技术集成到Chromebook Plus笔记本电脑中。这项技术此前已应用于谷歌的其他设备。华硕和惠普已经在市场上销售的Chromebook Plus机型&#xff0c;…...

Git Large File Storage (LFS) 的安装与使用

Git Large File Storage [LFS] 的安装与使用 1. An open source Git extension for versioning large files2. Installing on Linux using packagecloud3. Getting Started4. Error: Failed to call git rev-parse --git-dir: exit status 128References 1. An open source Git…...

使用国产工作流引擎,有那些好处?

使用国产工作流引擎的好处主要体现在以下几个方面&#xff1a; 符合企业独特业务&#xff1a; 国产工作流引擎可以深入挖掘和理解企业内部各项业务流程&#xff0c;精细化地定义流程模型和规则&#xff0c;实现“以流程驱动业务”的目标。这有助于企业更好地满足其独特的业务…...

掌握 Go 语言:使用 net/http/httptrace 包优化HTTP请求

掌握 Go 语言&#xff1a;使用 net/http/httptrace 包优化HTTP请求 介绍net/http/httptrace 包的基础概述适用场景 使用httptrace进行网络请求追踪配置httptrace的基本步骤示例&#xff1a;创建一个简单的HTTP客户端&#xff0c;使用httptrace监控连接 示例&#xff1a;追踪HTT…...

探秘Flask中的表单数据处理

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、引言 二、Flask中的表单处理机制 三、Flask表单处理实战 四、处理表单数据的注意事项…...

java —— 包装类及拆箱、装箱

java 当中有 8 种基本类型对应其相应的包装类&#xff0c;分别如下&#xff1a; intIntegerbyteByteshortShortlongLongfloatFloatdoubleDoublecharCharacterbooleanBoolean 一、装箱 两种装箱方法&#xff1a; public static void main(String[] args) {Integer anew Inte…...

运算符重载(下)

目录 前置和后置重载前置的实现Date& Date::operator()代码 后置的实现Date Date::operator(int )代码 前置--和后置--重载前置--的实现Date& Date::operator--( )代码 后置--的实现Date Date::operator--(int )代码 流插入运算符重载流插入运算符重载的实现流提取运算…...

杭州服务器的性能如何?

挥洒激情&#xff0c;开启杭州服务器的无限可能&#xff01; 互联网时代&#xff0c;服务器的性能就如同一艘航空母舰&#xff0c;承载着企业的发展梦想&#xff0c;指引着行业的发展方向。而对于杭州服务器&#xff0c;其性能究竟如何&#xff1f;让我来告诉您。 杭州服务器…...

linux centos nfs挂载两台服务器挂载统一磁盘目录权限问题

查看用户id id 用户名另一台为 修改uid和gid为相同id&#xff0c;添加附加组 usermod -u500 -Gwheel epms groupmod -g500 epms...

STL:string

文章目录 标准库中的string类string的构造string的赋值重载string的容量size(length)max_sizeresizereservecapacityclearemptyshink_to_fit string的元素访问operator[] 和 atfront 和 back string的迭代器 和 范围forstring的修改operatorappendpush_backassigninserterasere…...

贷款借钱平台 小额贷款系统开发小额贷款源码 贷款平台开发搭建

这款是贷款平台源码/卡卡贷源码/小贷源码/完美版 后台51800 密码51800 数据库替换application/database.php程序采用PHPMySQL&#xff0c;thinkphp框架代码开源&#xff0c;不加密后台效果&#xff1a;手机版效果 这款是贷款平台源码/卡卡贷源码/小贷源码/完美版 后台51800 密码…...

软设之算法的效率

算法的效率分为时间复杂度和空间复杂度。 空间复杂度是指对一个算法在运行过程中临时占用存储空间大小的度量。一个算法的空间复杂度只考虑在运行过程中为局部变量分配的存储空间的大小。说白了&#xff0c;就是空间换时间。 比如说计算从123……100的和。一个算法是i(1100)*…...

前端开发(2)--HTML常用的标签

100编程书屋_孔夫子旧书网 HTMl 的标签可以分为单个标签和成对标签。 单个标签&#xff1a;html4 规定单个标签要有一个 / 表示结尾&#xff0c; html5 则不用 <!--单个标签--> <meta> <!--成对标签 --> <div></div>以下是HTMl中常用的一些标签…...

任何图≌自己这一几何最起码常识推翻直线公理让R外标准实数一下子浮出水面

黄小宁 h定理&#xff1a;点集AB≌B的必要条件是A≌B。 证&#xff1a;若AB则A必可恒等变换地变为BA≌A&#xff0c;而恒等变换是保距变换。证毕。 如图所示R轴即x轴各元点x沿x轴正向不保距平移变为点y2x就使x轴沿本身拉伸&#xff08;放大&#xff09;变换为y2x轴不≌x轴&…...

js 纯前端实现数组分页、列表模糊查询、将数组转成formdata格式传给接口

后端返回所有的数据&#xff0c;由前端来实现分页展示、模糊查询&#xff0c;并将数组格式转成formdata格式给后端 1、数组转formdata let formData new FormData()for (let i 0; i < list.length; i) {let item list[i];for (let property in item) {formData.append(…...

elasticsearch有什么用

Elasticsearch是一个开源的分布式搜索和分析引擎&#xff0c;它被广泛用于构建实时的、可扩展的搜索和分析应用程序。以下是Elasticsearch的主要用途和功能&#xff1a;12 全文搜索&#xff1a;Elasticsearch提供强大的全文搜索功能&#xff0c;可以处理大量的文本数据&…...

iOS自动连接已知Wi-Fi功能的实现

首先需要在配置文件申请的时候将hotspot勾选上&#xff0c;之后还要在x-code里添加对应的配置&#xff0c;由于我们并没有用到获取设备周边Wi-Fi的功能&#xff0c;所以就没申请相关权限 相关连接Wi-Fi代码如下&#xff1a; #import <NetworkExtension/NetworkExtension.h&…...

编辑任何场景! 3DitScene:通过语言引导的解耦 Gaussian Splatting开源来袭!

文章&#xff1a;https://arxiv.org/pdf/2405.18424 项目&#xff1a;https://zqh0253.github.io/3DitScene/ huggingface:https://huggingface.co/spaces/qihang/3Dit-Scene 场景图像编辑在娱乐、摄影和广告设计中至关重要。现有方法仅专注于2D个体对象或3D全局场景编辑&…...

CCIG 2024:合合信息文档解析技术突破与应用前景

目录 背景当前大模型训练和应用面临的问题训练Token耗尽训练语料质量要求高LLM文档问答应用中文档解析不精准 合合信息的文档解析技术1. 具备多文档元素识别能力2. 具备版面分析能力3. 高性能的文档解析4. 高精准、高效率的文档解析文档多板式部分示例 文档解析典型技术难点元素…...

关于TeamSpeak3-网易音乐机器人的基础使用方法(胎教级教程)

本文转自博主的个人博客&#xff1a;https://blog.zhumengmeng.work,欢迎大家前往查看。 原文链接&#xff1a;点我访问 序言&#xff1a;在自己的ts服务器上安装了网易音乐机器人&#xff0c;写这篇文章旨在教群友/网友如何使用机器人!&#x1f60b;&#x1f44d; 一、TS3Audi…...

看广告赚金币提现小游戏app开发源码

开发一个看广告赚金币并可以提现的小游戏APP&#xff0c;源码的搭建涉及到多个方面&#xff0c;包括前端界面设计、后端逻辑处理、数据库管理以及广告平台的对接等。以下是一些建议的步骤和考虑因素&#xff1a; 前端界面设计&#xff1a; 使用HTML5、CSS3和JavaScript等技术…...

【vue】@、@/、../和./的区别

&#xff1a;表示vue语法中v-on的简写&#xff1b;绑定事件的专用格式。当事件触发的时候&#xff0c;函数才会来调用&#xff1b; /&#xff1a;在build文件夹下webpack.base.conf.js找到&#xff0c;便能知道代表什么了; 这里指向src文件夹 . /&#xff1a;表示当前目录下&…...

imx93 uboot 构建

1. 信息来源 从 nxp 的英文网站中可以找到 imx93 的构建信息&#xff0c;当前的最新版本为&#xff1a;Linux 6.6.3_1.0.0 # 网址如下&#xff1a; https://www.nxp.com/design/design-center/software/embedded-software/i-mx-software/embedded-linux-for-i-mx-application…...

视觉SLAM十四讲:从理论到实践(Chapter7:视觉里程计1)

前言 学习笔记&#xff0c;仅供学习&#xff0c;不做商用&#xff0c;如有侵权&#xff0c;联系我删除即可 一、目标 1.理解图像特征点的意义&#xff0c;并掌握在单幅图像中提取特征点及多幅图像中匹配特征点的方法。 2.理解对极几何的原理&#xff0c;利用对极几何的约束&…...

c++指针的*

1.*运算符 *运算符被称为间接值&#xff08;indirect value&#xff09;或者解除引用&#xff08;dereferencing&#xff09;运算符 将其应用于指针可以得到该地址处存储的值 由于 * 也有乘法的意思&#xff0c;c通过上下文来确定是乘法还是解除引用 2.如何声明和初始化指针 …...