0

0

C++怎么实现堆排序_C++大顶堆小顶堆构建【实战】

冰火之心

冰火之心

发布时间:2026-03-16 18:54:01

|

394人浏览过

|

来源于php中文网

原创

std::make_heap默认构建大顶堆,小顶堆需显式传std::greater();否则pop_heap等操作结果错误。大顶堆:std::make_heap(v.begin(), v.end());小顶堆:std::make_heap(v.begin(), v.end(), std::greater())。

c++怎么实现堆排序_c++大顶堆小顶堆构建【实战】

怎么用 std::make_heap 构建大顶堆和小顶堆

默认就是大顶堆,小顶堆必须显式传比较器,否则所有 pop_heappush_heap 行为都会出错——比如你以为在维护最小值,结果弹出来的是最大值。

常见错误现象:std::make_heap(v.begin(), v.end()) 后调用 std::pop_heap 取到的不是最小值;或者手动写循环建堆时根节点值不符合预期。

  • 大顶堆(默认):std::make_heap(v.begin(), v.end())
  • 小顶堆(必须):std::make_heap(v.begin(), v.end(), std::greater<int>())</int>
  • 自定义类型必须提供可比性,比如 struct Node { int val; }; bool operator,否则编译失败
  • 注意:std::greater<t></t> 要包含 <functional>,漏掉会报 error: 'greater' was not declared in this scope

为什么不能直接对 vector 排序完再调 make_heap

堆不是排序数组,它是满足偏序关系的完全二叉树结构。如果先 std::sortmake_heap,相当于拿一个有序序列强行“重排成堆”,不仅多余,还可能掩盖逻辑错误——比如你本意是动态维护 top-k,却误用了全量排序。

使用场景:需要频繁取极值 + 增删元素(如优先队列底层、滑动窗口最大值),而不是一次性求全局最值。

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

知我AI
知我AI

一款多端AI知识助理,通过一键生成播客/视频/文档/网页文章摘要、思维导图,提高个人知识获取效率;自动存储知识,通过与知识库聊天,提高知识利用效率。

下载
  • make_heap 时间复杂度是 O(n),比 n 次 push_heap 的 O(n log n) 更优,适合批量建堆
  • 但建好之后每次插入仍要 push_heap(O(log n)),不能靠 push_back + 再次 make_heap,那样是 O(n) 每次,性能崩盘
  • vector 尾部插入后必须立刻 push_heap,否则堆结构失效;同理,pop_heap 后要 pop_back 才真正删除

push_heappop_heap 的操作顺序不能颠倒

这两个函数只调整结构,不改变容器大小。很多人卡在这一步:调了 pop_heap 发现最后元素没变,以为失败了——其实它只是把堆顶换到了末尾,你还得手动 v.pop_back()

错误写法:std::pop_heap(v.begin(), v.end()); auto x = v.back(); —— 此时 v.back() 是原堆顶,但 v.size() 没变,下次 pop_heap 还会处理这个残留值。

  • 正确弹出最小值(小顶堆):std::pop_heap(v.begin(), v.end(), std::greater<int>()); auto min_val = v.back(); v.pop_back();</int>
  • 正确插入新值:v.push_back(x); std::push_heap(v.begin(), v.end(), std::greater<int>());</int>
  • 注意:所有 heap 算法都要求迭代器范围有效,v.empty() 时调 pop_heap 是未定义行为,务必判空

手写堆排序时,为什么下标从 0 开始比从 1 开始更容易出错

STL 容器索引从 0 开始,而经典教材讲堆常假设根在下标 1,子节点是 2i2i+1。直接套用会导致越界或父子关系错位,比如 parent = (i-1)/2 写成 i/2,在 i=0 时除零或算错。

实际开发中几乎没人手写完整堆排序——除非面试或嵌入式无 STL 环境。用 STL 就别碰裸下标计算。

  • STL 堆算法只依赖随机访问迭代器,跟具体下标无关,不用管内部怎么存
  • 如果真要手写,统一用 0 起始公式:parent = (i-1)/2left = 2*i+1right = 2*i+2,且必须检查 left < size
  • 递归式 sift_down 容易栈溢出,生产代码建议用 while 循环实现
堆排序的边界动作特别琐碎,比如 pop_heap 后忘 pop_back,或者比较器漏传,问题往往不报错而是逻辑错。这些点不像语法错误能被编译器拦住,得靠对每一步数据状态的明确预期来守住。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1071

2023.08.02

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

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

617

2024.08.29

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

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

335

2025.08.29

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

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

235

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

448

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

90

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

136

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

377

2026.03.11

热门下载

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

精品课程

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

共94课时 | 11.5万人学习

C 教程
C 教程

共75课时 | 5.5万人学习

C++教程
C++教程

共115课时 | 22.2万人学习

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

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