0

0

c++怎么实现二叉堆的向上调整与向下调整_c++ 优先队列底层逻辑【详解】

冰火之心

冰火之心

发布时间:2026-01-03 10:33:33

|

759人浏览过

|

来源于php中文网

原创

向上调整需确保i>0再算父节点索引,用有符号int防绕回;向下调整须先判left

c++怎么实现二叉堆的向上调整与向下调整_c++ 优先队列底层逻辑【详解】

向上调整 heapify_up 怎么写才不越界

向上调整发生在插入新元素后,把新节点从堆底一路“浮”到合适位置。关键不是“怎么比较”,而是**索引计算必须防越界**——父节点索引是 (i - 1) / 2,但当 i == 0(已在根)时必须立即停止。

常见错误是写成 while (i > 0 && arr[i] > arr[(i-1)/2]) 却忘了在循环体内更新 i 后没检查新 i 是否仍合法;更隐蔽的坑是用无符号整数(如 size_t i),i - 1 会绕回极大值,直接崩。

  • 始终用有符号整型(如 int)作下标变量
  • 循环条件严格写成 i > 0,且更新 i 后不依赖后续判断来兜底
  • 比较前不假设父节点存在——i > 0 就是唯一安全前提
void heapify_up(vector& heap, int i) {
    while (i > 0) {
        int parent = (i - 1) / 2;
        if (heap[i] <= heap[parent]) break; // 小顶堆:子不小于父则停
        swap(heap[i], heap[parent]);
        i = parent;
    }
}

向下调整 heapify_down 的三个子节点边界判断

向下调整用于弹出堆顶或修改根后,把顶部元素“沉”到底部。难点不在逻辑,而在**左右子节点索引是否越界**:左子为 2*i + 1,右子为 2*i + 2,但这两个索引随时可能 ≥ heap.size()

不能只写 if (left heap[largest]) 就完事——如果 left 已越界,right 就不该参与比较;更糟的是,有些实现先算 right 再判断,导致 right 计算本身越界(尤其用无符号类型)。

立即学习C++免费学习笔记(深入)”;

  • 先算 left = 2*i + 1,若 left >= size,直接退出(无子节点)
  • 再算 right = left + 1,仅当 right 才和 left 比较选大者
  • 选中的最大子节点必须和当前节点比较,且 swap 后要继续向下,不能漏掉深层调整
void heapify_down(vector& heap, int i, int size) {
    while (true) {
        int largest = i;
        int left = 2 * i + 1;
        if (left < size && heap[left] > heap[largest]) largest = left;
        int right = left + 1;
        if (right < size && heap[right] > heap[largest]) largest = right;
        if (largest == i) break;
        swap(heap[i], heap[largest]);
        i = largest;
    }
}

std::priority_queue 底层真用 vector + 手动调整吗

是的,std::priority_queue 默认容器是 std::vector,默认比较器是 std::less(大顶堆),所有插入、弹出都靠隐式调用 push_heap / pop_heap,而这两个函数底层就是你写的 heapify_upheapify_down 的泛型版本。

RoomGPT
RoomGPT

使用AI为每个人创造梦想的房间

下载

但它不直接暴露调整函数——你无法对已有 vector 调用 make_heap 后再手动调用 push_heap,除非你用原始数组或迭代器区间。实际项目中,如果需要中途修改某个元素并重平衡,priority_queue 无能为力,必须自己维护 vector + 下标映射 + 手动调整。

  • priority_queuetop()O(1)push()pop() 都是 O(log n)
  • 它不提供“降低/升高某元素优先级”的接口,这是很多算法(如 Dijkstra)必须手写堆的原因
  • 调试时可把内部 c 成员(底层容器)用 const_cast 强转出来观察,但别修改——破坏堆序会导致后续操作未定义行为

小顶堆 vs 大顶堆:改一个参数还是重写两套逻辑

别重写。C++ 标准库靠比较器控制堆序:priority_queue, greater> 是小顶堆,less 是大顶堆。自己实现时也应统一用模板比较器,而不是硬编码 >

真正容易错的是:以为把所有 > 换成 就行——其实向上/向下调整中“违反堆序”的判定方向必须一致。例如小顶堆要求父 ≤ 子,那么 heapify_up 中就要写 if (heap[i] >= heap[parent]) break;,不是简单翻转符号。

  • 把比较逻辑抽成独立函数对象或 lambda,传入调整函数
  • 测试时务必用混杂正负数的用例,避免因全正/全负掩盖符号错误
  • 注意 std::greater 对自定义类型需重载 operator>,而 std::lessoperator,别搞反

真实场景里,最常被忽略的是调整过程中对容器 size 的动态信任——比如在 heapify_down 里传入的 size 如果不是当前有效长度(例如删了元素但没 shrink),就会访问野指针;或者多线程环境下没加锁,size() 和后续下标访问之间被 resize 了。这些不会报编译错误,但 core dump 很安静。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
Sass和less的区别
Sass和less的区别

Sass和less的区别有语法差异、变量和混合器的定义方式、导入方式、运算符的支持、扩展性等。本专题为大家提供Sass和less相关的文章、下载、课程内容,供大家免费下载体验。

198

2023.10.12

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

713

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

82

2023.09.25

java中break的作用
java中break的作用

本专题整合了java中break的用法教程,阅读专题下面的文章了解更多详细内容。

116

2025.10.15

java break和continue
java break和continue

本专题整合了java break和continue的区别相关内容,阅读专题下面的文章了解更多详细内容。

254

2025.10.24

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

313

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

522

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

150

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Rust 教程
Rust 教程

共28课时 | 4万人学习

Git 教程
Git 教程

共21课时 | 2.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号