【Leetcode 热题 100】215. 数组中的第K个最大元素
问题配景给定整数数组 n u m s nums nums 和整数 k k k,请返回数组中第 k k k 个最大的元素。
请注意,你需要找的是数组排序后的第 k k k 个最大的元素,而不是第 k k k 个不同的元素。
你必须计划并实现时间复杂度为 O ( n ) O(n) O(n) 的算法办理此问题。
数据约束
[*] 1 ≤ k ≤ n u m s . l e n g t h ≤ 1 0 5 1 \le k \le nums.length \le 10 ^ 5 1≤k≤nums.length≤105
[*] − 1 0 4 ≤ n u m s [ i ] ≤ 1 0 4 -10 ^ 4 \le nums \le 10 ^ 4 −104≤nums≤104
解题过程
经典求第 k k k大,最佳方案是用随机选择算法,时间复杂度大致是 O ( N ) O(N) O(N) 这个量级。
考虑到标题分类在堆,那除了调用 API 快速处理的方法之外,再写一下手撕堆看成练习。
干系详细的先容,可以参考 随机快排与随机选择 以及 堆排序与堆操作。
具体实现
使用堆 API
class Solution {
public int findKthLargest(int[] nums, int k) {
Queue<Integer> queue = new PriorityQueue<>();
for(int num : nums) {
queue.offer(num);
}
for(int i = 0; i < nums.length - k; i++) {
queue.poll();
}
return queue.peek();
}
}
自己实现堆
class Solution {
public int findKthLargest(int[] nums, int k) {
int n = nums.length;
buildHeap(nums);
// 注意这里要修正索引
while(k - 1 > 0) {
swap(nums, 0, --n);
downAdjust(nums, 0, n);
k--;
}
return nums;
}
// 交换数组中的两个元素
private void swap(int[] arr, int i, int j) {
int temp = arr;
arr = arr;
arr = temp;
}
// 由上而下调整元素
private void downAdjust(int[] arr, int cur, int size) {
// 数组下标从零开始,当前节点的左孩子下标为 2 * cur + 1
int child = 2 * cur + 1;
while (child < size) {
// 如果当前节点有右孩子,那么比较两个子节点的值确定潜在的交换对象
int target = child + 1 < size && arr > arr ? child + 1 : child;
// 再与当前节点比较大小
target = arr > arr ? target : cur;
// 一旦发现此次操作中无需交换,立即停止流程
if (target == cur) {
break;
}
// 交换父子节点
swap(arr, target, cur);
// 移动工作指针
cur = target;
child = 2 * cur + 1;
}
}
// 自底向顶建堆
private void buildHeap(int[] arr) {
int n = arr.length;
for (int i = n - 1; i >= 0; i--) {
downAdjust(arr, i, n);
}
}
}
随机选择
class Solution {
// 全局变量 first 表示等于当前元素的第一个下标位置,last 表示最后一个
private static int first, last;
public int findKthLargest(int[] nums, int k) {
return randomizedSelect(nums, nums.length - k);
}
// 交换数组中的两个元素
private void swap(int[] arr, int i, int j) {
int temp = arr;
arr = arr;
arr = temp;
}
private void partition(int[] arr, int left, int right, int pivot) {
// 初始化维护的指针
first = left;
last = right;
int cur = left;
// 注意循环到工作指针超越 last 即可结束流程
while (cur <= last) {
if (arr == pivot) {
cur++; // 当前元素等于划分标准,不作处理并后移指针
} else if (arr < pivot) {
swap(arr, first++, cur++); // 小于划分标准,交换且分别后移两个指针
} else {
swap(arr, cur, last--); // 大于划分标准,交换并前移记录最后一个当前元素的指针
}
}
}
private int randomizedSelect(int[] arr, int target) {
int res = 0;
for (int left = 0, right = arr.length - 1; left <= right;) {
partition(arr, left, right, arr);
if (target < first) {
right = first - 1; // 当前下标小于维护的第一个下标,到左边区间中找
} else if (target > last) {
left = last + 1; // 当前下标大于维护的最后一个下标,到右边区间中找
} else {
res = arr; // 当前下标在维护的范围内,记录结果
break;
}
}
return res;
}
}
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页:
[1]