0

0

C++怎么实现堆排序_C++优先队列教程【高效】

穿越時空

穿越時空

发布时间:2026-03-04 13:27:12

|

363人浏览过

|

来源于php中文网

原创

堆排序在c++中应优先使用std::make_heap、std::pop_heap、std::sort_heap三步完成,而非手写堆;std::priority_queue默认为最大堆,需显式指定std::greater实现最小堆;手写堆需注意索引公式、边界检查与sift_down顺序。

c++怎么实现堆排序_c++优先队列教程【高效】

堆排序的核心不是手写堆,而是用 std::make_heap 三步到位

堆排序在 C++ 里真没必要从 sift_down 开始造轮子。标准库已经提供完整支持,关键在于理解三步操作的语义和顺序:std::make_heapstd::pop_heapstd::sort_heap。它们不是独立函数,而是一套协作协议——先建堆,再逐个弹出最大值到尾部,最后原地有序。

常见错误是把 std::pop_heap 当成“删除最大值”,其实它只是把堆顶换到末尾、然后重排剩余部分,容器大小不变。必须配合 vector.pop_back() 才算真正移除;否则下一次调用会越界或逻辑错乱。

  • std::make_heap(v.begin(), v.end()):默认建最大堆(升序排序用)
  • 循环调用 std::pop_heap(v.begin(), v.end() - i)(i 是已弹出个数),再执行 v.pop_back()
  • 如果只想要排序结果,直接 std::sort_heap(v.begin(), v.end()) ——但前提是它已经是合法堆

优先队列 std::priority_queue 默认是最大堆,别被名字骗了

std::priority_queue 名字叫“优先队列”,行为却是取最大值,这和多数人直觉相反(比如任务调度常要最小截止时间)。它底层默认用 std::less<t></t>,所以 top() 返回最大元素。想变成最小堆?得显式传比较器:

std::priority_queue<int, std::vector<int>, std::greater<int>> min_heap;

注意三个模板参数缺一不可:元素类型、容器类型、比较器。漏掉 std::vector<int></int> 会导致编译失败,错误信息类似 "no type named 'value_type' in 'std::less<int>'"</int>

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

Pixelfox AI
Pixelfox AI

多功能AI图像编辑工具

下载
  • 插入用 push(),取顶用 top(),弹出用 pop()(不返回值)
  • size()empty() 安全,但没有 find() 或遍历接口——它不是容器,是适配器
  • 性能上,push()pop() 都是 O(log n),但内存局部性不如 vector 原生堆操作

手写堆时最容易崩在索引计算和边界条件上

如果你非得实现一个最小堆类,90% 的 bug 出在两个地方:父/子节点索引公式写反,以及 sift_down 里没处理“只有一个子节点”的情况。数组下标从 0 开始时,左子节点是 2*i + 1,右子是 2*i + 2,父节点是 (i-1)/2(整除)。别用 i/2,那适用于下标从 1 开始的教材伪代码。

另一个坑是 sift_down 过程中,比较前必须先检查子节点是否越界。例如判断右子是否存在,不能只写 right ,还要确保 <code>left 已成立,否则 <code>right 可能是负数或溢出。

  • 建堆用 sift_down 从最后一个非叶子节点开始(索引 (size-2)/2),不是从 0
  • 插入新元素用 sift_up,从末尾往上比,比父小就交换(最小堆)
  • C++ 中用 std::swap(a[i], a[parent]) 比手动赋值更安全,避免自赋值问题

std::make_heapstd::priority_queue 的内存与缓存差异

两者都基于完全二叉树,但内存布局不同:std::make_heap 直接操作原始 vector,数据连续,CPU 缓存友好;std::priority_queue 是对容器的封装,虽然底层也是 vector,但每次 push 可能触发多次内存拷贝(尤其存对象时)。如果元素类型大且移动代价高,建议用 emplace() 替代 push(),避免临时对象构造。

兼容性方面,std::make_heap 要求迭代器是随机访问,std::list 不行;std::priority_queue 第二个模板参数强制要求支持 push_back 和随机访问(所以 deque 可以,list 不行)。

  • 大数据量排序优先用 std::make_heap + std::sort_heap,少一次内存分配
  • 动态增删场景用 std::priority_queue,但注意它不支持修改已有元素优先级
  • 需要改优先级?要么用 std::set 模拟,要么自己维护索引映射,别硬刚堆结构

堆结构看着简单,但索引偏移、边界检查、比较器语义、内存模型这四点串起来,就是容易静默出错的地方。动手前先想清楚:你要的是排序结果,还是持续的动态最大值查询。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

215

2023.10.12

string转int
string转int

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

930

2023.08.02

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

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

600

2024.08.29

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

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

294

2025.08.29

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

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

212

2025.08.29

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

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

1794

2023.10.19

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

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

572

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2340

2025.12.29

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

0

2026.03.04

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.6万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 20.3万人学习

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

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