建堆时间复杂度

片头

嗨!小伙伴们,大家好! 在上一篇中,我们学习了什么是堆,以及如何实现堆。这一篇中,我将继续带领大家来深入学习堆,准备好了吗?我要开始咯!

首先,大家还记得这个代码吗?

//堆的初始化
void HeapInit(Heap* hp) {
	assert(hp);		//断言,防止传入空指针
	hp->arr = NULL; //动态数组为空
	hp->capacity = 0;//初始时,数组的容量为0
	hp->size = 0;	 //初始时,数组的大小为0
}

哈哈,是的!这就是我们初始化堆的代码,那我们今天同样是初始化堆,但是思路和之前不太一样哦!

首先,我们定义一个接口函数 HeapCreate,里面包含3个参数,分别是 Heap* hp, ElemType* arr, int n,函数的返回类型为 void

void HeapCreate(Heap* hp,ElemType* a,int n);

三个参数的意义如下:

  • Heap* hp:  指向堆这个结构体的指针,用来对堆进行修改操作
  •  ElemType* arr: 指向外界提供数组的指针
  • int n: n表示要在arr数组拷贝的元素个数

关于这个接口函数,具体实现代码如下:

//堆的构建
void HeapCreate(Heap* hp, ElemType* a, int n) {
	assert(hp);            //断言,防止传入空指针
	assert(a);             //断言,防止传入空指针

//对堆这个结构体的成员变量arr开辟一块空间(堆的本质就是数组)
	hp->arr = (ElemType*)malloc(sizeof(ElemType)*n);//开辟n个大小的空间
	if (hp->arr == NULL) {    //如果内存空间不足
		perror("malloc fail!\n");
		exit(1);
	}
	memcpy(hp->arr, a, n * sizeof(ElemType));//将数组a的所有元素拷贝到堆的arr中
	hp->capacity = n;//初始化堆的容量为n
	hp->size = n;    //初始化堆的元素个数为n

	//建堆
	//向上调整建堆
	/*for (int i = 1; i < hp->size; i++) {        //从第二个结点开始挪动
		AdjustUp(hp->arr, i);
	}*/
	//向下调整建堆
	for (int i = (hp->size - 1 - 1) / 2; i >= 0; i--) {    //从倒数第一个非叶子结点开始挪动
		AdjustDown(hp->arr, hp->size, i);
	}
}

测试代码如下:

内容我们很熟悉,但是同样都是建堆,向上调整建堆和向下调整建堆有什么差异呢? 它们两个进行比较,哪一个算法更优一点呢?

不急,且听我慢慢道来~

我们先来看看向上调整建堆

我们以满二叉树来举例子,因为它是一种特殊的二叉树,其中每个非叶子节点都有两个子节点,并且所有叶子节点都在同一层上。

二叉树的每一层结点数目如下:

我们现在采用向上调整建堆的思想,创建的第一个结点可以看成小堆(大堆),因此,我们不需要对第一个结点作任何处理。创建第二个结点,我们就需要向上调整1次,创建第三个结点,我们就需要向上调整1次,因此,第二层总共有2个结点,我们需要调整1次,也就是 2^1*1。

因此,设累和向上调整F(H), F(H) = F(h1)+F(h2)+F(h3)+F(h4)+.......+F(h-1)+F(h) ,其中,F(h) = 每层结点个数 * 向上调整次数 

所以,F(H) = 2^1*1 + 2^2*2 + 2^3*3 + ...... + 2^(h-2)*(h-2) + 2^(h-1)*(h-1) ,我们用错位相减法来求解此题

 2*F(H) = 2^2*1 + 2^3*2 + 2^4*3 + ...... + 2^(h-2)*(h-3)+ 2^(h-1)*(h-2) + 2^(h)*(h-1) 

F(H) = 2^1*1 + 2^2*2 + 2^3*3 + 2^4*4 + ....... + 2^(h-2)*(h-2) + 2^(h-1)*(h-1) 

将两个式子相减:

 2*F(H) - F(H) = -2^1 - 2^2  - 2^3 - 2^4 -  ....... - 2^(h-2) - 2^(h-1) + 2^(h)*(h-1)

我们在等式中补充:-2^0 + 2^0 ,原式变成:-2^1 - 2^2  - 2^3 - 2^4 -  ....... - 2^(h-2) - 2^(h-1) - 2^0 + 2^0 + 2^(h)*(h-1),调换一下顺序,就变成下面这个样子:

 2*F(H) - F(H)  =   - 2^0 - 2^1 - 2^2  - 2^3 - 2^4 -  ....... - 2^(h-2) - 2^(h-1) + 2^0 + 2^(h)*(h-1)

 F(H) = - (2^0 +  2^1 + 2^2 + 2^3 + 2^4 +.......+ 2^(h-2) +  2^(h-1) )+ 2^0 + 2^(h)*(h-1)

【其中,2^0 +  2^1 + 2^2 + 2^3 + 2^4 +.......+ 2^(h-2) +  2^(h-1),可以进行错位相减,设N是树中结点的数量,N = 2^0 + 2^1 + 2^2 + 2^3 + 2^4 +.....+ 2^(h-2) + 2^(h-1) 

   2*N =  2^1 + 2^2 + 2^3 + 2^4 + 2^5 +.....+ 2^(h-2) + 2^(h-1) + 2^(h) 

    2*N - N =  2^(h) - 2^0 = 2^h - 1 

    N  =  2^h - 1, 2^h = N+1, h = log(N+1)    (注意:本来是以2为底数,N+1为真数 ,但是也可以省略底数】 

因此,上述可以化简为:F(H) = - (2^h-1) + 2^0 + 2^(h)*(h-1)  =  -2^h + 1 + 1 + 2^h*(h-1)                F(H)  = 2^h*(h-2) + 2, 将 F(H) 转换成 F(N) , 2^h = N+1,  h =  log(N+1) , F(N) =(N+1)*(log(N+1) -2) +2,F(N) = N*logN , 因此,向上调整建堆的时间复杂度为 O(N*logN)。


好啦,向上调整建堆的时间复杂度我们算出来了,那么向下调整建堆的时间复杂度怎么求呢?

还是以满二叉树来举例:

我们需要向下调整,建堆,最后一层(第h层)的叶子结点可以看作(大堆/小堆),所以我们不挪动叶子结点,因此,从倒数第一个非叶子结点开始挪动(也就是最后一个叶子结点的父节点),也就是第h-1层开始挪动,第h-1层的每一个结点最多向下调整1次,第h-2层的每一个结点最多向下调整2次,第h-3层的每一个结点最多向下调整3次,.........., 第5层的每一个结点最多向下调整h-5次,第4层的每一个结点最多向下调整h-4次,第3层的每一个结点最多向下调整h-3次,第2层的每一个结点最多向下调整h-2次,第1层的每一个结点最多向下调整h-1次。

因此,设累和向下调整F(H), F(H) = F(h1)+F(h2)+F(h3)+F(h4)+.......+F(h-1)+F(h) ,其中,F(h) = 每层结点个数 * 向下调整次数 

所以,F(H) = 2^(h-2)*1 + 2^(h-3)*2 + ...... + 2^3*(h-4) + 2^2*(h-3) + 2^1*(h-2) + 2^0*(h-1),我们用错位相减法来求解此题

 F(H) = 2^(h-2)*1 + 2^(h-3)*2 + ...... + 2^3*(h-4) + 2^2*(h-3) + 2^1*(h-2) + 2^0*(h-1)

2*F(H) = 2^(h-1)*1 + 2^(h-2)*2 + ...... + 2^4*(h-4) + 2^3*(h-3) + 2^2*(h-2) + 2^1*(h-1)

将两个式子相减可得

 2*F(H) -  F(H)  =  2^(h-1)*1 + 2^(h-2)*1 + 2^(h-3)*1 +......+ 2^4 + 2^+ 2^2+ 2^1 - 2^0*(h-1)

F(H)  = 2^(h-1)*1 + 2^(h-2)*1 + 2^(h-3)*1 +......+ 2^4 + 2^+ 2^2+ 2^1 - (h-1)

F(H)  = 2^(h-1)*1 + 2^(h-2)*1 + 2^(h-3)*1 +......+ 2^4 + 2^+ 2^+ 2^1 - h + 1 (这个“+1”可以看成 2^0), 我们可以调换一下顺序:

F(H)  = 2^0 + 2^+ 2^2 + 2^+ 2^4 +......+ 2^(h-3)*1 + 2^(h-2)*1 + 2^(h-1)*1 - h

【其中,2^0 +  2^1 + 2^2 + 2^3 + 2^4 +.......+ 2^(h-2) +  2^(h-1),可以进行错位相减,设N是树中结点的数量,  N  = 2^0 + 2^1 + 2^2 + 2^3 + 2^4 +.....+ 2^(h-2) + 2^(h-1) 

 2*N =  2^1 + 2^2 + 2^3 + 2^4 + 2^5 +.....+ 2^(h-2) + 2^(h-1) + 2^(h) 

 2*N - N =  2^(h) - 2^0 = 2^h - 1 

 N  =  2^h - 1, 2^h = N+1, h = log(N+1)    (注意:本来是以2为底数,N+1为真数 ,但是也可以省略底数】 

因此,上述可以化简为:F(H) = 2^h - 1 - h, 我们将 F(H) 转换为 F(N) , F(N) = (N+1) -1 - log(N+1)  , F(N) = N - log(N+1) , 因此,向下调整建堆的时间复杂度为 O(N)。


向上调整建堆的时间复杂度为 O(N*logN) ,向下调整建堆的时间复杂度为 O(N)。 那肯定是向下调整建堆的时间复杂度更小, 因此我们会选择向下调整建堆。

哈哈哈,还有一种简便方法,那就是直接看图,不需要计算~  你想想,假设树的高度为h, 满二叉树的最后一层叶子结点有多少个? 前面我们画图分析过,最后一层叶子结点有 2^(h-1) 个, 那么整棵二叉树的结点总数有多少? 设N是树中结点的数量, N  =  2^h - 1,我们可以看到, 2^(h-1) 和2^h 只相差2倍, 换句话说,最后一层的叶子结点占结点总数的1/2,如果我们采用向上调整算法,相当于整棵二叉树一半的结点都要挪动,是不是很费时? 而如果我们采用向下调整算法,最后一层的叶子结点保持不动,只需要移动上面的结点,是不是要轻松很多?

设累和调整F(H), F(H) = F(h1)+F(h2)+F(h3)+F(h4)+.......+F(h-1)+F(h) ,其中,F(h) = 每层结点个数 * 调整次数    , 采用向上调整算法, 相当于 每层结点个数多 * 调整次数多(简称: 多*多),采用向下调整算法,相当于 每层结点个数多 * 调整次数少 或者 每层结点个数少 * 调整次数多(简称: 少*多 或者 多*少)

分析以上情况,我们可以得出:向下调整算法优于向上调整算法,所以如果要建堆,尽量选择向下调整算法。

片尾

今天我们学习了建堆的时间复杂度,希望看完这篇文章能对友友们有所帮助 ! ! !

点赞收藏加关注 !  !  !

谢谢大家 !  !  !

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

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

相关文章

opencv_17_翻转与旋转

一、图像翻转 1&#xff09;void flip_test(Mat& image); 2&#xff09;void ColorInvert::flip_test(Mat& image) { Mat dst; //flip(image, dst, 0); //上下翻转 flip(image, dst, 1); //左右翻转 // flip(image, dst, -1); //180度翻转 imsho…

VScode 无法连接云服务器

试了很多方法&#xff0c;比如更换VScode版本&#xff0c;卸载重装&#xff0c;删除配置文件 重启电脑&#xff0c;都无法成功。最后重置电脑后才连接上&#xff0c;但是重启服务器后又出现该问题。 方法一&#xff1a;修改环境 方法二&#xff1a;把vscode卸载干净重下

【快速入门】数据库的增删改查与结构讲解

文章的操作都是基于小皮php study的MySQL5.7.26进行演示 what 数据库是能长期存储在计算机内&#xff0c;有组织的&#xff0c;可共享的大量数据的集合。数据库中的数据按照一定的数据模型存储&#xff0c;具有较小的冗余性&#xff0c;较高的独立性和易扩展性&#xff0c;并为…

LabVIEW智能变电站监控系统设计与实现

LabVIEW智能变电站监控系统设计与实现 随着电力系统和智能化技术的快速发展&#xff0c;建立一个高效、可靠的变电站监控系统显得尤为重要。通过分析变电站监控系统的需求&#xff0c;设计了一个基于LabVIEW软件的监控平台。该平台利用虚拟仪器技术、传感器技术和无线传输技术…

数据结构中的栈(C语言版)

一.栈的概念 栈是一种常见的数据结构&#xff0c;它遵循后进先出的原则。栈可以看作是一种容器&#xff0c;其中的元素按照一种特定的顺序进行插入和删除操作。 压栈&#xff1a;栈的插入操作叫做进栈/压栈/入栈&#xff0c;入数据在栈顶。 出栈&#xff1a;栈的删除操作叫做…

2024年的十大技术趋势 - AI 等等

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

网优小工具-基站ID行列转换

网优小工具&#xff0d;基站ID行列转换 因在日常工作需要对基站ID批量行转列&#xff0c;以方便在网管上批量筛选指定网元&#xff0c;该小工具基于微软Power Query插件编写&#xff0c;工具方便、简洁、易用&#xff0c;共享出来以方便工作。 工作界面 &#xff11;.粘贴需筛…

学习VUE2第6天

一.请求拦截器 可以节流&#xff0c;防止多次点击请求 toast是单例 二.前置路由守卫 在Vue.js中&#xff0c;前置路由守卫是指在路由转换实际发生之前执行的钩子函数。这是Vue Router&#xff08;Vue.js官方的路由管理器&#xff09;提供的一种功能&#xff0c;允许开发者在用…

中兴UME网管LTE共享参数配置-PLMN添加

本文为中兴设备UME网管电联中频共享参数配置&#xff0c;PLMN添加参数配置部分&#xff0c;因UME与U&#xff13;&#xff11;网管添加PLMN配置区别较大&#xff0c;UME网管需同时配置运营商EN&#xff0d;DC策略&#xff0c;相关配置流程及参数配置如下文。 PLMN eNodeB CU …

与 Apollo 共创生态:观看7周年大会的心路历程

前言 在科技飞速发展的今天&#xff0c;自动驾驶技术已然成为行业创新的热点之一。作为一名长期关注自动驾驶领域的技术人员&#xff0c;我有幸见证了Apollo平台的成长与壮大。七年前&#xff0c;Apollo的诞生为我们带来了无尽的想象与期待&#xff1b;七年后的今天&#xff0…

【自研网关系列】过滤器链 -- 灰度发布过滤器

&#x1f308;Yu-Gateway&#xff1a;&#xff1a;基于 Netty 构建的自研 API 网关&#xff0c;采用 Java 原生实现&#xff0c;整合 Nacos 作为注册配置中心。其设计目标是为微服务架构提供高性能、可扩展的统一入口和基础设施&#xff0c;承载请求路由、安全控制、流量治理等…

【介绍下Unity编辑器扩展】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

ubuntu下安装配置python3.11

方案1 添加仓库&#xff1a; $ sudo add-apt-repository ppa:deadsnakes/ppa $ sudo apt update $ sudo apt install python3.11然后查看有多少个python版本已经安装了&#xff1a; ls -l /usr/bin/python*python2.7,python 3.8 ,python 3.11. 然后&#xff0c;设置系统默认…

Android4.4真机移植过程笔记(一)

1、RK源码编译 获取内核源码&#xff1a; git clone git172.28.1.172:rk3188_kernel -b xtc_ok1000 内核编译环境&#xff1a; 从172.28.1.132编译服务器的/data1/ZouZhiPing目录下拷贝toolchain.tar.gz&#xff08;交叉编译工具链&#xff09;并解压到与rk3188_kernel同级目…

Visual Studio中怎样更改Nuget程序包源

场景 Visual Studio 2019 在使用NuGet添加依赖包时&#xff0c;在预览中搜索不到程序包。 排查下NuGet的程序包源为本地。 将程序包源修改下。 实现 在解决方案上右击选择管理解决方案中的NuGet程序包(在 Visual Studio 中打开“工具”>“选项”>“NuGet 包管理器”…

【配置】Docker搭建JSON在线解析网站

云服务器打开端口8787 连接上docker运行 docker run -id --name jsonhero -p 8787:8787 -e SESSION_SECRETabc123 henryclw/jsonhero-webhttp://ip:8787访问 Github&#xff1a;地址

考研数学|《880题》不会做,怎么办?

如果880大部分都不会做&#xff0c;说明基础掌握太差&#xff0c;如果已经到了10月底&#xff0c;我建议直接刷知能行吧。因为我一个同学经历和你类似&#xff0c;最后通过一个月高强度刷知能行算是补救了一些。 对于刚开始准备考研的同学和上面的题主&#xff0c;我想聊几句建…

PCIe协议之RCB、MPS、MRRS详解

✨前言&#xff1a; PCIe总线的存储器写请求、存储器读完成等TLP中含有数据负载&#xff0c;即Data Payload。Data Payload的长度和MPS&#xff08;Max Payload Size&#xff09;、MRRS&#xff08;Max Read Request Size&#xff09;和RCB&#xff08;Read Completion Bounda…

物联网实战--平台篇之(二)基础搭建

目录 一、Qt工程创建 二、数据库知识 三、通信协议 四、名词定义 本项目的交流QQ群:701889554 物联网实战--入门篇https://blog.csdn.net/ypp240124016/category_12609773.html 物联网实战--驱动篇https://blog.csdn.net/ypp240124016/category_12631333.html 一、Qt工程…

请编写函数fun,该函数的功能是:统计各年龄段的人数。N个年龄通过调用随机函数获得,并放在主函数的age数组中;

本文收录于专栏:算法之翼 https://blog.csdn.net/weixin_52908342/category_10943144.html 订阅后本专栏全部文章可见。 本文含有题目的题干、解题思路、解题思路、解题代码、代码解析。本文分别包含C语言、C++、Java、Python四种语言的解法完整代码和详细的解析。 题干 请编…
最新文章