news 2026/4/24 18:23:14

堆是一种特殊的完全二叉树结构,用于高效实现优先队列

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
堆是一种特殊的完全二叉树结构,用于高效实现优先队列

堆是一种特殊的完全二叉树结构,用于高效实现优先队列。其基本性质如下:

  1. 结构性质:堆是一棵完全二叉树,可以用数组紧凑存储,无空洞。

    • 对于数组下标从 0 开始的情况:
      • 节点i的父节点下标为(i-1)//2
      • 左孩子下标为2*i+1,右孩子为2*i+2
  2. 堆序性质

    • 大根堆(最大堆):任意节点的值 ≥ 其左右子节点的值,根节点为最大值。
    • 小根堆(最小堆):任意节点的值 ≤ 其左右子节点的值,根节点为最小值。
  3. 建堆过程(自底向上调整)

    • 从最后一个非叶子节点开始,即第⌊n/2⌋ - 1个元素(0起始索引),向前依次调用HeapAdjust函数进行向下调整,直到根节点。
    • 时间复杂度为 O(n),优于逐个插入的 O(n log n)。
  4. HeapAdjust(调整函数,以大根堆为例)

defheap_adjust(data,s,m):# 在 data[s..m] 中,仅 data[s] 可能不满足堆性质,其余已满足temp=data[s]child=2*s+1# 左孩子whilechild<=m:# 找出较大的子节点ifchild<manddata[child]<data[child+1]:child+=1# 如果根不小于子节点,则已合适iftemp>=data[child]:break# 否则将较大子节点上移data[s]=data[child]s=child child=2*s+1data[s]=temp# 插入原 data[s] 到正确位置
  1. HeapSort(堆排序算法)
defheap_sort(data):n=len(data)# 构建大根堆:从最后一个非叶节点开始调整foriinrange(n//2-1,-1,-1):heap_adjust(data,i,n-1)# 逐个提取堆顶并重建堆foriinrange(n-1,0,-1):data[0],data[i]=data[i],data[0]# 堆顶与末尾交换heap_adjust(data,0,i-1)# 重新调整剩余元素成堆
  1. 性能分析
  • 时间复杂度:O(n log n) —— 建堆 O(n),每次调整 O(log n),共 n−1 次。
  • 空间复杂度:O(1),原地排序。
  • 稳定性:不稳定,相同元素可能因父子交换而改变相对顺序。
  • 适用场景
    • 数据量大且对最坏情况有要求(相比快排更稳定);
    • Top-K 问题(如找最大前 K 个数);
    • 实时系统中需要可预测性能的排序任务。
    • 将一个数组原地构建成小根堆的过程与构建大根堆类似,核心思想是利用完全二叉树的结构特性自底向上的调整策略。只需在调整函数中将比较方向反转(父节点 ≤ 子节点),即可实现小根堆。

步骤说明:

  1. 存储方式:使用数组表示完全二叉树,索引从 0 开始。

    • 节点i的左孩子为2*i+1,右孩子为2*i+2
    • 父节点为(i-1)//2
  2. 关键点:最后一个非叶子节点的下标是⌊n/2⌋ - 1(因为叶子节点没有子节点)。

  3. 调整顺序:从该节点开始,向前逐个调用min_heap_adjust函数进行“向下筛选”。


Python 实现代码

defmin_heap_adjust(data,s,m):""" 调整 data[s] 使得子树 data[s..m] 满足小根堆性质 前提:data[s] 的左右子树已经是小根堆 """temp=data[s]child=2*s+1# 左孩子whilechild<=m:# 选择较小的子节点ifchild<manddata[child]>data[child+1]:child+=1# 如果当前值已经不大于子节点,则位置合适iftemp<=data[child]:break# 否则将较小的子节点上移data[s]=data[child]s=child child=2*s+1data[s]=temp# 将原值放入正确位置defbuild_min_heap(data):""" 将数组 data 原地构建成小根堆 """n=len(data)# 从最后一个非叶子节点开始向前调整foriinrange(n//2-1,-1,-1):min_heap_adjust(data,i,n-1)# 示例使用arr=[4,10,3,5,1]build_min_heap(arr)print("小根堆:",arr)# 输出如: [1, 4, 3, 5, 10]

特点分析

  • 时间复杂度:O(n),虽然单次调整为 O(log n),但由于树的层次分布特性,整体建堆为线性。
  • 空间复杂度:O(1),原地操作。
  • 稳定性:堆排序本身不稳定,但建堆过程不影响稳定性讨论(排序时才体现)。

通过上述方法,可以高效地将任意数组原地转换为一个小根堆,适用于最小优先队列、Top-K 最小元素等问题。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 13:31:54

从内存占用到执行速度:C#中List、ImmutableArray、ReadOnlyCollection真实性能测评

第一章&#xff1a;C#数据集合性能测评综述在开发高性能 .NET 应用程序时&#xff0c;选择合适的数据集合类型对整体性能具有决定性影响。不同的集合类型在插入、删除、查找和遍历等操作中表现出显著差异&#xff0c;合理评估其性能特征有助于优化内存使用和执行效率。常见集合…

作者头像 李华
网站建设 2026/4/23 11:15:12

如何用C#实现真正可复用的业务模块?这3种模式必须掌握

第一章&#xff1a;Shell脚本的基本语法和命令Shell脚本是Linux/Unix系统中自动化任务的核心工具&#xff0c;它允许用户通过一系列命令的组合来执行复杂的操作。编写Shell脚本时&#xff0c;通常以“shebang”开头&#xff0c;用于指定解释器。脚本的起始声明 所有Shell脚本应…

作者头像 李华
网站建设 2026/4/23 11:15:38

国际志愿者组织:灾区物资清单OCR识别协调救援分配

国际志愿者组织&#xff1a;灾区物资清单OCR识别协调救援分配 在一场突如其来的地震过后&#xff0c;临时搭建的救援指挥帐篷里&#xff0c;志愿者正焦急地翻看一叠手写和打印混杂的物资清单——“矿泉水 300箱”、“奶粉 45罐”、“毛毯 200条”……这些信息需要尽快录入系统&…

作者头像 李华
网站建设 2026/4/23 10:07:19

清华镜像站资源更新:腾讯混元OCR国内高速下载通道上线

清华镜像站上线腾讯混元OCR国内高速下载通道&#xff1a;轻量高效&#xff0c;一键部署 在文档数字化浪潮席卷各行各业的今天&#xff0c;一个现实问题始终困扰着开发者——如何快速、准确地从一张扫描发票、身份证或复杂排版的PDF中提取出结构化信息&#xff1f;传统OCR方案虽…

作者头像 李华
网站建设 2026/4/22 21:06:49

医疗文档处理难点破解:腾讯混元OCR支持病历扫描件结构化解析

医疗文档处理难点破解&#xff1a;腾讯混元OCR支持病历扫描件结构化解析 在医院档案室里&#xff0c;成堆的纸质病历静静躺在文件柜中。这些承载着患者诊疗信息的重要资料&#xff0c;却因缺乏数字化手段而难以被有效利用。每当需要调取历史记录时&#xff0c;医护人员往往要花…

作者头像 李华
网站建设 2026/4/23 20:47:06

电商平台打假:商品详情页截图OCR比对正品参数差异

电商平台打假&#xff1a;商品详情页截图OCR比对正品参数差异 在电商平台上&#xff0c;你有没有遇到过这样的情况——图片上写着“iPhone 15原装充电器”&#xff0c;点进去却发现是个山寨品牌&#xff1f;或者看到某款手机标注“6.8英寸OLED屏、支持5G”&#xff0c;结果一查…

作者头像 李华