建堆时间复杂度

2024-04-27 19:12
文章标签 复杂度 时间 建堆

本文主要是介绍建堆时间复杂度,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

片头

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

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

//堆的初始化
void HeapInit(Heap* hp) {assert(hp);		//断言,防止传入空指针hp->arr = NULL; //动态数组为空hp->capacity = 0;//初始时,数组的容量为0hp->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;//初始化堆的容量为nhp->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.chinasem.cn/article/941304

相关文章

服务器集群同步时间手记

1.时间服务器配置(必须root用户) (1)检查ntp是否安装 [root@node1 桌面]# rpm -qa|grep ntpntp-4.2.6p5-10.el6.centos.x86_64fontpackages-filesystem-1.41-1.1.el6.noarchntpdate-4.2.6p5-10.el6.centos.x86_64 (2)修改ntp配置文件 [r

MiniGPT-3D, 首个高效的3D点云大语言模型,仅需一张RTX3090显卡,训练一天时间,已开源

项目主页:https://tangyuan96.github.io/minigpt_3d_project_page/ 代码:https://github.com/TangYuan96/MiniGPT-3D 论文:https://arxiv.org/pdf/2405.01413 MiniGPT-3D在多个任务上取得了SoTA,被ACM MM2024接收,只拥有47.8M的可训练参数,在一张RTX

批处理以当前时间为文件名创建文件

批处理以当前时间为文件名创建文件 批处理创建空文件 有时候,需要创建以当前时间命名的文件,手动输入当然可以,但是有更省心的方法吗? 假设我是 windows 操作系统,打开命令行。 输入以下命令试试: echo %date:~0,4%_%date:~5,2%_%date:~8,2%_%time:~0,2%_%time:~3,2%_%time:~6,2% 输出类似: 2019_06

【MRI基础】TR 和 TE 时间概念

重复时间 (TR) 磁共振成像 (MRI) 中的 TR(重复时间,repetition time)是施加于同一切片的连续脉冲序列之间的时间间隔。具体而言,TR 是施加一个 RF(射频)脉冲与施加下一个 RF 脉冲之间的持续时间。TR 以毫秒 (ms) 为单位,主要控制后续脉冲之前的纵向弛豫程度(T1 弛豫),使其成为显著影响 MRI 中的图像对比度和信号特性的重要参数。 回声时间 (TE)

LeetCode:64. 最大正方形 动态规划 时间复杂度O(nm)

64. 最大正方形 题目链接 题目描述 给定一个由 0 和 1 组成的二维矩阵,找出只包含 1 的最大正方形,并返回其面积。 示例1: 输入: 1 0 1 0 01 0 1 1 11 1 1 1 11 0 0 1 0输出: 4 示例2: 输入: 0 1 1 0 01 1 1 1 11 1 1 1 11 1 1 1 1输出: 9 解题思路 这道题的思路是使用动态规划

O(n)时间内对[0..n^-1]之间的n个数排序

题目 如何在O(n)时间内,对0到n^2-1之间的n个整数进行排序 思路 把整数转换为n进制再排序,每个数有两位,每位的取值范围是[0..n-1],再进行基数排序 代码 #include <iostream>#include <cmath>using namespace std;int n, radix, length_A, digit = 2;void Print(int *A,

算法复杂度的简单介绍

算法复杂度是衡量算法执行效率和资源消耗的指标,通常分为时间复杂度和空间复杂度。时间复杂度评估算法执行所需时间随输入规模的变化,空间复杂度评估算法占用内存的增长情况。复杂度通常用大O符号来表示,它描述了最坏情况下的增长速率。 1. 时间复杂度 时间复杂度表示算法执行所需时间随输入规模 nnn 的变化关系。常见的时间复杂度如下(从快到慢): a. 常数时间:O(1) 不管输入大小如何,算法总是

LeetCode:3177. 求出最长好子序列 II 哈希表+动态规划实现n*k时间复杂度

3177. 求出最长好子序列 II 题目链接 题目描述 给你一个整数数组 nums 和一个非负整数k 。如果一个整数序列 seq 满足在下标范围 [0, seq.length - 2] 中 最多只有 k 个下标i满足 seq[i] != seq[i + 1] ,那么我们称这个整数序列为好序列。请你返回 nums中好子序列的最长长度。 实例1: 输入:nums = [1,2,1,1,3],

未雨绸缪:环保专包二级资质续期工程师招聘时间策略

对于环保企业而言,在二级资质续期前启动工程师招聘的时间规划至关重要。考虑到招聘流程的复杂性、企业内部需求的变化以及政策标准的更新,建议环保企业在二级资质续期前至少提前6至12个月启动工程师招聘工作。这个时间规划可以细化为以下几个阶段: 一、前期准备阶段(提前6-12个月) 政策与标准研究: 深入研究国家和地方关于环保二级资质续期的最新政策、法规和标准,了解对工程师的具体要求。评估政策变化可

用Python实现时间序列模型实战——Day 14: 向量自回归模型 (VAR) 与向量误差修正模型 (VECM)

一、学习内容 1. 向量自回归模型 (VAR) 的基本概念与应用 向量自回归模型 (VAR) 是多元时间序列分析中的一种模型,用于捕捉多个变量之间的相互依赖关系。与单变量自回归模型不同,VAR 模型将多个时间序列作为向量输入,同时对这些变量进行回归分析。 VAR 模型的一般形式为: 其中: ​ 是时间  的变量向量。 是常数向量。​ 是每个时间滞后的回归系数矩阵。​ 是误差项向量,假