给你一个整数数组 nums 和一个整数 k ,请你返回其中出现频率前 k 高的元素。你可以按 任意顺序 返回答案。
示例 1:
输入: nums = [1,1,1,2,2,3], k = 2
输出: [1,2]
示例 2:
输入: nums = [1], k = 1
输出: [1]
来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/top-k-frequent-elements
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解法:
- 先用Map记录每个数字出现的次数;
- 使用优先级队列(大顶堆)对Map中的元素进行排序 (只保留最大的k个元素)
class Solution {
public int[] topKFrequent(int[] nums, int k) {
// 把每个元素出现的频次记录到map中
Map<Integer, Integer> map = new HashMap<>();
for (int i = 0; i < nums.length; i++){
map.put(nums[i], map.getOrDefault(nums[i], 0)+1);
}
// 优先级队列,使用数组的第二个元素进行比较。
// 这里的数组第一位是元素值,第二位是元素出现的次数
PriorityQueue<int[]> queue = new PriorityQueue<>(new Comparator<int[]>() {
@Override
public int compare(int[] o1, int[] o2) {
return o1[1] - o2[1];
}
});
// 遍历Map,把每一对 元素和次数组成的数组,放到优先级队列中
Set<Map.Entry<Integer, Integer>> entrySet = map.entrySet();
for (Map.Entry<Integer, Integer> entry : entrySet) {
int num = entry.getKey();
int times = entry.getValue();
if (queue.size() == k) {
if (queue.peek()[1] < times) {
queue.poll(); // 这里弹出的是最小值
queue.offer(new int[]{num, times});// times会自动排到peek位置
}
}else {
queue.offer(new int[]{num, times});
}
}
// 取出出现频次最多的元素并返回
int result[] = new int[k];
for (int i = 0; i < k; i++) {
result[i] = queue.poll()[0];
}
return result;
}
}