0

0

c++中如何实现堆排序_c++堆排序算法逻辑

尼克

尼克

发布时间:2026-01-17 13:15:45

|

511人浏览过

|

来源于php中文网

原创

堆排序核心是手动在数组上维护最大堆性质,分建堆(O(n))和逐次取最大值排序(O(n log n))两步,需正确计算下标、从最后一个非叶子节点开始建堆并缩小堆边界。

c++中如何实现堆排序_c++堆排序算法逻辑

堆排序的核心是维护最大堆性质

堆排序不是靠调用现成的堆容器,而是手动在数组上模拟完全二叉树结构,通过反复调整使根节点始终为子树最大值。关键在于理解 heapify 过程:从最后一个非叶子节点开始向上 sift-down,确保以每个节点为根的子树满足最大堆(父 ≥ 左右子)。

常见错误是下标计算出错——C++ 数组从 0 开始,节点 i 的左子为 2*i+1,右子为 2*i+2,父节点为 (i-1)/2。漏掉这个偏移会导致越界或逻辑错乱。

分两步实现:建堆 + 逐个取出最大值

建堆阶段是 O(n),不是直觉上的 n×log n,因为大部分节点在底层、调整代价小;排序阶段每次把堆顶(最大)与末尾交换,再对剩余 n-1 元素重新 heapify 根节点,共 n-1 次,每次 O(log n)

  • 建堆必须从最后一个非叶子节点开始,即 index = n/2 - 1(向下取整),不是从 0 或 n-1
  • 交换后要缩小堆边界(heap_size--),否则会把已排好的最大值再次参与比较
  • heapify 内部需递归或循环下滤,不能只比一次就停——可能需要一直沉到底

一个可直接运行的 C++ 实现示例

以下代码不依赖 STL 堆,纯数组操作,含注释说明关键位置:

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

PaperAiBye
PaperAiBye

支持近30多种语言降ai降重,并且支持多种语言免费测句子的ai率,支持英文aigc报告等

下载
void heapify(std::vector& arr, int n, int i) {
    int largest = i;
    int left = 2 * i + 1;
    int right = 2 * i + 2;
if (left zuojiankuohaophpcn n && arr[left] youjiankuohaophpcn arr[largest])
    largest = left;
if (right zuojiankuohaophpcn n && arr[right] youjiankuohaophpcn arr[largest])
    largest = right;

if (largest != i) {
    std::swap(arr[i], arr[largest]);
    heapify(arr, n, largest); // 继续下滤
}

}

void heapSort(std::vector& arr) { int n = arr.size();

// 步骤1:建最大堆(从最后一个非叶子节点开始)
for (int i = n / 2 - 1; i youjiankuohaophpcn= 0; i--)
    heapify(arr, n, i);

// 步骤2:逐个取出堆顶,放至末尾
for (int i = n - 1; i youjiankuohaophpcn 0; i--) {
    std::swap(arr[0], arr[i]); // 最大值移到已排序区
    heapify(arr, i, 0);        // 对剩余 i 个元素重新堆化
}

}

容易被忽略的边界和性能细节

堆排序是原地、不稳定排序,heapify 的递归写法虽清晰但有开销;生产环境常用迭代版避免深递归。另外:std::make_heapstd::sort_heap标准库封装,但它们底层仍是同样逻辑,只是接口更简洁。

真正难调试的点往往在:n=0n=1 时建堆循环是否跳过、right 下标是否越界、heapify 后未更新 largest 就继续用旧值比较——这些都会导致部分数据不参与排序或结果错乱。

相关文章

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

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

下载

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

相关专题

更多
string转int
string转int

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

317

2023.08.02

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

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

538

2024.08.29

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

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

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

97

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1022

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

64

2025.10.17

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.16

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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