215. Kth Largest Element in an Array (M)
https://leetcode.com/problems/kth-largest-element-in-an-array/
Given an integer array nums
and an integer k
, return the kth
largest element in the array.
Note that it is the kth
largest element in the sorted order, not the kth
distinct element.
Example 1:
Example 2:
Constraints:
1 <= k <= nums.length <= 104
-104 <= nums[i] <= 104
Solution:
解法一
二叉堆的解法比较简单,实际写算法题的时候,推荐大家写这种解法,先直接看代码吧:
二叉堆(优先队列)是比较常见的数据结构,可以认为它会自动排序,我们前文 手把手实现二叉堆数据结构 实现过这种结构,我就默认大家熟悉它的特性了。
看代码应该不难理解,可以把小顶堆 pq
理解成一个筛子,较大的元素会沉淀下去,较小的元素会浮上来;当堆大小超过 k
的时候,我们就删掉堆顶的元素,因为这些元素比较小,而我们想要的是前 k
个最大元素嘛。
当 nums
中的所有元素都过了一遍之后,筛子里面留下的就是最大的 k
个元素,而堆顶元素是堆中最小的元素,也就是「第 k
个最大的元素」。
二叉堆插入和删除的时间复杂度和堆中的元素个数有关,在这里我们堆的大小不会超过 k
,所以插入和删除元素的复杂度是 O(logK)
,再套一层 for 循环,总的时间复杂度就是 O(NlogK)
。空间复杂度很显然就是二叉堆的大小,为 O(K)
。
这个解法算是比较简单的吧,代码少也不容易出错,所以说如果笔试面试中出现类似的问题,建议用这种解法。唯一注意的是,Java 的 PriorityQueue
默认实现是小顶堆,有的语言的优先队列可能默认是大顶堆,可能需要做一些调整。
解法二
快速选择算法比较巧妙,时间复杂度更低,是快速排序的简化版,一定要熟悉思路。
我们先从快速排序讲起。
快速排序的逻辑是,若要对 nums[lo..hi]
进行排序,我们先找一个分界点 p
,通过交换元素使得 nums[lo..p-1]
都小于等于 nums[p]
,且 nums[p+1..hi]
都大于 nums[p]
,然后递归地去 nums[lo..p-1]
和 nums[p+1..hi]
中寻找新的分界点,最后整个数组就被排序了。
快速排序的代码如下:
关键就在于这个分界点索引 p
的确定,我们画个图看下 partition
函数有什么功效:
索引 p
左侧的元素都比 nums[p]
小,右侧的元素都比 nums[p]
大,意味着这个元素已经放到了正确的位置上,回顾快速排序的逻辑,递归调用会把 nums[p]
之外的元素也都放到正确的位置上,从而实现整个数组排序,这就是快速排序的核心逻辑。
那么这个 partition
函数如何实现的呢?看下代码:
熟悉快速排序逻辑的读者应该可以理解这段代码的含义了,这个 partition
函数细节较多,上述代码参考《算法4》,是众多写法中最漂亮简洁的一种,所以建议背住,这里就不展开解释了。
好了,对于快速排序的探讨到此结束,我们回到一开始的问题,寻找第 k
大的元素,和快速排序有什么关系?
注意这段代码:
我们刚说了,partition
函数会将 nums[p]
排到正确的位置,使得 nums[lo..p-1] < nums[p] < nums[p+1..hi]
。
那么我们可以把 p
和 k
进行比较,如果 p < k
说明第 k
大的元素在 nums[p+1..hi]
中,如果 p > k
说明第 k
大的元素在 nums[lo..p-1]
中。
所以我们可以复用 partition
函数来实现这道题目,不过在这之前还是要做一下索引转化:
题目要求的是「第 k
个最大元素」,这个元素其实就是 nums
升序排序后「索引」为 len(nums) - k
的这个元素。
这样就可以写出解法代码:
这个代码框架其实非常像我们前文 二分搜索框架 的代码,这也是这个算法高效的原因,但是时间复杂度为什么是 O(N)
呢?按理说类似二分搜索的逻辑,时间复杂度应该一定会出现对数才对呀?
其实这个 O(N)
的时间复杂度是个均摊复杂度,因为我们的 partition
函数中需要利用 双指针技巧 遍历 nums[lo..hi]
,那么总共遍历了多少元素呢?
最好情况下,每次 p
都恰好是正中间 (lo + hi) / 2
,那么遍历的元素总数就是:
N + N/2 + N/4 + N/8 + … + 1
这就是等比数列求和公式嘛,求个极限就等于 2N
,所以遍历元素个数为 2N
,时间复杂度为 O(N)
。
但我们其实不能保证每次 p
都是正中间的索引的,最坏情况下 p
一直都是 lo + 1
或者一直都是 hi - 1
,遍历的元素总数就是:
N + (N - 1) + (N - 2) + … + 1
这就是个等差数列求和,时间复杂度会退化到 O(N^2)
,为了尽可能防止极端情况发生,我们需要在算法开始的时候对 nums
数组来一次随机打乱:
前文 洗牌算法详解 写过随机乱置算法,这里就不展开了。当你加上这段代码之后,平均时间复杂度就是 O(N)
了,提交代码后运行速度大幅提升。
总结一下,快速选择算法就是快速排序的简化版,复用了 partition
函数,快速定位第 k
大的元素。相当于对数组部分排序而不需要完全排序,从而提高算法效率,将平均时间复杂度降到 O(N)
。
Last updated