动态求前n个最小值(最大值)
注: 由于最小值和最大值的分析过程完全相同,这里我们只讨论最小值的分析流程,最大值同理
问题描述
每次给定一个数值,询问此数值以及之前给定数值中最小的n
个数
例如给定数值的顺序为:8 7 1 2 9 4
,设n == 3
8
:不足3
个数,则答案为8
7
:不足3
个数,则答案为8 7
1
:不足3
个数,则答案为8 7 1
2
:8 7 1 2
中较小的3个数为7 1 2
9
:8 7 1 2 9
中较小的3个数为7 1 2
4
:8 7 1 2 9 4
中较小的3个数为4 1 2
暴力做法
一个最直观的方法是,每次添加新的数据后,将当前所有数据进行排序,选择较小的n个
一次排序最快也要,总的复杂度至少要
小根堆和大根堆
维护一个容量为n
的大根堆,堆内元素代表当前最小的n
个值
每次添加数值与堆顶元素进行比较,具有以下两种情况:
- 大于堆顶元素: 由于是大根堆,堆顶元素为最小的
n
个数中的最大值,则此数值一定大于n个数中的其他值,即它一定不属于答案 - 小于堆顶元素:则用它取代堆顶元素则可以使得这
n
个数更小
对于问题描述中所举的例子,使用大根堆的求解流程如下:
8
:不足3
个数,则答案为8
7
:不足3
个数,则答案为8 7
1
:不足3
个数,则答案为8 7 1
2
:2 < 8
, 则答案为7 2 1
9
:9 > 7
,则答案不变为7 2 1
4
:4 < 7
,则答案为4 2 1
总结
动态求前n
个最小值使用大根堆
动态求前n
个最大值使用小根堆
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 终于写完轮子一部分:tcp代理 了,记录一下
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理