|
| 1 | +### 题目描述 |
| 2 | + |
| 3 | +这是 LeetCode 上的 **[215. 数组中的第K个最大元素](https://leetcode.cn/problems/kth-largest-element-in-an-array/solution/by-ac_oier-x9h4/)** ,难度为 **中等**。 |
| 4 | + |
| 5 | +Tag : 「树状数组」、「二分」、「优先队列(堆)」、「快速选择」 |
| 6 | + |
| 7 | + |
| 8 | + |
| 9 | +给定整数数组 `nums` 和整数 `k`,请返回数组中第 `k` 个最大的元素。 |
| 10 | + |
| 11 | +请注意,你需要找的是数组排序后的第 `k` 个最大的元素,而不是第 `k` 个不同的元素。 |
| 12 | + |
| 13 | +你必须设计并实现时间复杂度为 $O(n)$ 的算法解决此问题。 |
| 14 | + |
| 15 | +示例 1: |
| 16 | +``` |
| 17 | +输入: [3,2,1,5,6,4], k = 2 |
| 18 | +
|
| 19 | +输出: 5 |
| 20 | +``` |
| 21 | +示例 2: |
| 22 | +``` |
| 23 | +输入: [3,2,3,1,2,4,5,5,6], k = 4 |
| 24 | +
|
| 25 | +输出: 4 |
| 26 | +``` |
| 27 | + |
| 28 | +提示: |
| 29 | +* $1 <= k <= nums.length <= 10^5$ |
| 30 | +* $-10^4 <= nums[i] <= 10^4$ |
| 31 | + |
| 32 | +--- |
| 33 | + |
| 34 | +### 值域映射 + 树状数组 + 二分 |
| 35 | + |
| 36 | +除了直接对数组进行排序,取第 $k$ 位的 $O(n\log{n})$ 做法以外。 |
| 37 | + |
| 38 | +对于值域大小 小于 数组长度本身时,我们还能使用「树状数组 + 二分」的 $O(n\log{m})$ 做法,其中 $m$ 为值域大小。 |
| 39 | + |
| 40 | +首先值域大小为 $[-10^4, 10^4]$,为了方便,我们为每个 $nums[i]$ 增加大小为 $1e4 + 10$ 的偏移量,将值域映射到 $[10, 2 \times 10^4 + 10]$ 的空间。 |
| 41 | + |
| 42 | +将每个增加偏移量后的 $nums[i]$ 存入树状数组,考虑在 $[0, m)$ 范围内进行二分,假设我们真实第 $k$ 大的值为 $t$,那么在以 $t$ 为分割点的数轴上,具有二段性质: |
| 43 | + |
| 44 | +* 在 $[0, t]$ 范围内的数 $cur$ 满足「树状数组中大于等于 $cur$ 的数不低于 $k$ 个」 |
| 45 | +* 在 $(t, m)$ 范围内的数 $cur$ 不满足「树状数组中大于等于 $cur$ 的数不低于 $k$ 个」 |
| 46 | + |
| 47 | +二分出结果后再减去刚开始添加的偏移量即是答案。 |
| 48 | + |
| 49 | +代码: |
| 50 | +```Java |
| 51 | +class Solution { |
| 52 | + int M = 10010, N = 2 * M; |
| 53 | + int[] tr = new int[N]; |
| 54 | + int lowbit(int x) { |
| 55 | + return x & -x; |
| 56 | + } |
| 57 | + int query(int x) { |
| 58 | + int ans = 0; |
| 59 | + for (int i = x; i > 0; i -= lowbit(i)) ans += tr[i]; |
| 60 | + return ans; |
| 61 | + } |
| 62 | + void add(int x) { |
| 63 | + for (int i = x; i < N; i += lowbit(i)) tr[i]++; |
| 64 | + } |
| 65 | + public int findKthLargest(int[] nums, int k) { |
| 66 | + for (int x : nums) add(x + M); |
| 67 | + int l = 0, r = N - 1; |
| 68 | + while (l < r) { |
| 69 | + int mid = l + r + 1 >> 1; |
| 70 | + if (query(N - 1) - query(mid - 1) >= k) l = mid; |
| 71 | + else r = mid - 1; |
| 72 | + } |
| 73 | + return r - M; |
| 74 | + } |
| 75 | +} |
| 76 | +``` |
| 77 | +* 时间复杂度:将所有数字放入树状数组复杂度为 $O(n\log{m})$;二分出答案复杂度为 $O(\log^2{m})$,其中 $m = 2 \times 10^4$ 为值域大小。整体复杂度为 $O(n\log{m})$ |
| 78 | +* 空间复杂度:$O(m)$ |
| 79 | + |
| 80 | +--- |
| 81 | + |
| 82 | +### 优先队列(堆) |
| 83 | + |
| 84 | +另外一个容易想到的想法是利用优先队列(堆),由于题目要我们求的是第 $k$ 大的元素,因此我们建立一个小根堆。 |
| 85 | + |
| 86 | +根据当前队列元素个数或当前元素与栈顶元素的大小关系进行分情况讨论: |
| 87 | + |
| 88 | +* 当优先队列元素不足 $k$ 个,可将当前元素直接放入队列中; |
| 89 | +* 当优先队列元素达到 $k$ 个,并且当前元素大于栈顶元素(栈顶元素必然不是答案),可将当前元素放入队列中。 |
| 90 | + |
| 91 | +代码: |
| 92 | +```Java |
| 93 | +class Solution { |
| 94 | + public int findKthLargest(int[] nums, int k) { |
| 95 | + PriorityQueue<Integer> q = new PriorityQueue<>((a,b)->a-b); |
| 96 | + for (int x : nums) { |
| 97 | + if (q.size() < k || q.peek() < x) q.add(x); |
| 98 | + if (q.size() > k) q.poll(); |
| 99 | + } |
| 100 | + return q.peek(); |
| 101 | + } |
| 102 | +} |
| 103 | +``` |
| 104 | +* 时间复杂度:$O(n\log{k})$ |
| 105 | +* 空间复杂度:$O(k)$ |
| 106 | + |
| 107 | +--- |
| 108 | + |
| 109 | +### 快速选择 |
| 110 | + |
| 111 | +对于给定数组,求解第 $k$ 大元素,且要求线性复杂度,正解为使用「快速选择」做法。 |
| 112 | + |
| 113 | +基本思路与「快速排序」一致,每次敲定一个基准值 `x`,根据当前与 `x` 的大小关系,将范围在 $[l, r]$ 的 $nums[i]$ 划分为到两边。 |
| 114 | + |
| 115 | +同时利用,利用题目只要求输出第 $k$ 大的值,而不需要对数组进行整体排序,我们只需要根据划分两边后,第 $k$ 大数会落在哪一边,来决定对哪边进行递归处理即可。 |
| 116 | + |
| 117 | +> 快速排序模板为面试向重点内容,需要重要掌握。 |
| 118 | +
|
| 119 | +代码: |
| 120 | +```Java |
| 121 | +class Solution { |
| 122 | + int[] nums; |
| 123 | + int qselect(int l, int r, int k) { |
| 124 | + if (l == r) return nums[k]; |
| 125 | + int x = nums[l], i = l - 1, j = r + 1; |
| 126 | + while (i < j) { |
| 127 | + do i++; while (nums[i] < x); |
| 128 | + do j--; while (nums[j] > x); |
| 129 | + if (i < j) swap(i, j); |
| 130 | + } |
| 131 | + if (k <= j) return qselect(l, j, k); |
| 132 | + else return qselect(j + 1, r, k); |
| 133 | + } |
| 134 | + void swap(int i, int j) { |
| 135 | + int c = nums[i]; |
| 136 | + nums[i] = nums[j]; |
| 137 | + nums[j] = c; |
| 138 | + } |
| 139 | + public int findKthLargest(int[] _nums, int k) { |
| 140 | + nums = _nums; |
| 141 | + int n = nums.length; |
| 142 | + return qselect(0, n - 1, n - k); |
| 143 | + } |
| 144 | +} |
| 145 | +``` |
| 146 | +* 时间复杂度:期望 $O(n)$ |
| 147 | +* 空间复杂度:忽略递归带来的额外空间开销,复杂度为 $O(1)$ |
| 148 | + |
| 149 | +--- |
| 150 | + |
| 151 | +### 最后 |
| 152 | + |
| 153 | +这是我们「刷穿 LeetCode」系列文章的第 `No.215` 篇,系列开始于 2021/01/01,截止于起始日 LeetCode 上共有 1916 道题目,部分是有锁题,我们将先把所有不带锁的题目刷完。 |
| 154 | + |
| 155 | +在这个系列文章里面,除了讲解解题思路以外,还会尽可能给出最为简洁的代码。如果涉及通解还会相应的代码模板。 |
| 156 | + |
| 157 | +为了方便各位同学能够电脑上进行调试和提交代码,我建立了相关的仓库:https://github.com/SharingSource/LogicStack-LeetCode 。 |
| 158 | + |
| 159 | +在仓库地址里,你可以看到系列文章的题解链接、系列文章的相应代码、LeetCode 原题链接和其他优选题解。 |
| 160 | + |
0 commit comments