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/17 23:13:57

从零开始搭建OCR系统:使用腾讯HunyuanOCR进行端到端识别

从零开始搭建OCR系统&#xff1a;使用腾讯HunyuanOCR进行端到端识别 在文档数字化浪潮席卷各行各业的今天&#xff0c;企业每天面对成千上万张扫描件、发票、合同和截图&#xff0c;如何高效提取其中的文字信息&#xff1f;传统OCR方案往往需要部署多个模型——先检测文字位置&…

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

C#跨平台方法拦截全攻略(从入门到高级拦截技术大揭秘)

第一章&#xff1a;C#跨平台方法拦截概述 在现代软件开发中&#xff0c;C#已不再局限于Windows平台。随着.NET Core和.NET 5的统一&#xff0c;C#实现了真正的跨平台能力&#xff0c;能够在Linux、macOS等操作系统上运行。在此背景下&#xff0c;方法拦截&#xff08;Method In…

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

从解决“有没有”的规模追赶期,进入回答“好不好、强不强、新不新”的高质量发展攻坚期。

目录 一、核心趋势&#xff1a;一场面向未来的系统性重塑 二、重点研究方向&#xff1a;聚焦关键瓶颈与未来高地 三、实施路径建议&#xff1a;从战略到行动的桥梁 未来5-10年&#xff0c;中国轨道交通将完成从“世界领先的规模”到“世界领先的质量”的关键一跃。发展的核心…

作者头像 李华
网站建设 2026/4/18 10:13:45

你还在用foreach遍历百万级数据?:3个高效替代方案实测对比

第一章&#xff1a;Shell脚本的基本语法和命令Shell脚本是Linux/Unix系统中自动化任务的核心工具&#xff0c;通过编写可执行的文本文件&#xff0c;用户能够组合系统命令、控制程序流程并处理数据。编写Shell脚本的第一步是声明解释器&#xff0c;通常在脚本首行使用shebang&a…

作者头像 李华
网站建设 2026/4/22 16:44:56

揭秘C#自定义集合中的表达式奥秘:如何实现高性能数据查询

第一章&#xff1a;C#自定义集合与表达式树概述 在现代C#开发中&#xff0c;理解自定义集合和表达式树是构建高效、可扩展应用程序的关键。它们不仅增强了代码的灵活性&#xff0c;还为LINQ查询、动态逻辑构建提供了底层支持。 自定义集合的核心作用 允许开发者根据业务需求实…

作者头像 李华
网站建设 2026/4/18 20:48:48

模块间通信难题全解析,深度解读C#系统解耦最佳实践

第一章&#xff1a;模块间通信难题全解析&#xff0c;深度解读C#系统解耦最佳实践 在现代软件架构中&#xff0c;模块化设计已成为提升可维护性与扩展性的核心手段。然而&#xff0c;随着模块数量增加&#xff0c;模块间的通信复杂度急剧上升&#xff0c;紧耦合问题频发&#x…

作者头像 李华