feat: add top_k.c and refactor top_k.js (#889)

* Add top_k.c based on my_heap.c

* Improve the implementation of top_k.js

* Add a comment to top_k
This commit is contained in:
Yudong Jin 2023-10-26 02:54:19 +08:00 committed by GitHub
parent 9f4076d1c1
commit 7822bf9cd4
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 190 additions and 73 deletions

View file

@ -1,2 +1,2 @@
add_executable(my_heap my_heap.c) add_executable(my_heap_test my_heap_test.c)
add_executable(top_k top_k.c)

View file

@ -34,6 +34,12 @@ MaxHeap *newMaxHeap(int nums[], int size) {
return h; return h;
} }
/* 析构函数 */
void freeMaxHeap(MaxHeap *h) {
// 释放内存
free(h);
}
/* 获取左子节点索引 */ /* 获取左子节点索引 */
int left(MaxHeap *h, int i) { int left(MaxHeap *h, int i) {
return 2 * i + 1; return 2 * i + 1;
@ -144,37 +150,3 @@ void siftUp(MaxHeap *h, int i) {
i = p; i = p;
} }
} }
/* Driver Code */
int main() {
/* 初始化堆 */
// 初始化大顶堆
int nums[] = {9, 8, 6, 6, 7, 5, 2, 1, 4, 3, 6, 2};
MaxHeap *heap = newMaxHeap(nums, sizeof(nums) / sizeof(int));
printf("输入数组并建堆后\n");
printHeap(heap->data, heap->size);
/* 获取堆顶元素 */
printf("\n堆顶元素为 %d\n", peek(heap));
/* 元素入堆 */
push(heap, 7);
printf("\n元素 7 入堆后\n");
printHeap(heap->data, heap->size);
/* 堆顶元素出堆 */
int top = pop(heap);
printf("\n堆顶元素 %d 出堆后\n", top);
printHeap(heap->data, heap->size);
/* 获取堆大小 */
printf("\n堆元素数量为 %d\n", size(heap));
/* 判断堆是否为空 */
printf("\n堆是否为空 %d\n", isEmpty(heap));
// 释放内存
free(heap);
return 0;
}

View file

@ -0,0 +1,41 @@
/**
* File: my_heap_test.c
* Created Time: 2023-01-15
* Author: Reanon (793584285@qq.com)
*/
#include "my_heap.c"
/* Driver Code */
int main() {
/* 初始化堆 */
// 初始化大顶堆
int nums[] = {9, 8, 6, 6, 7, 5, 2, 1, 4, 3, 6, 2};
MaxHeap *heap = newMaxHeap(nums, sizeof(nums) / sizeof(int));
printf("输入数组并建堆后\n");
printHeap(heap->data, heap->size);
/* 获取堆顶元素 */
printf("\n堆顶元素为 %d\n", peek(heap));
/* 元素入堆 */
push(heap, 7);
printf("\n元素 7 入堆后\n");
printHeap(heap->data, heap->size);
/* 堆顶元素出堆 */
int top = pop(heap);
printf("\n堆顶元素 %d 出堆后\n", top);
printHeap(heap->data, heap->size);
/* 获取堆大小 */
printf("\n堆元素数量为 %d\n", size(heap));
/* 判断堆是否为空 */
printf("\n堆是否为空 %d\n", isEmpty(heap));
// 释放内存
freeMaxHeap(heap);
return 0;
}

View file

@ -0,0 +1,73 @@
/**
* File: top_k.c
* Created Time: 2023-10-26
* Author: Krahets (krahets163.com)
*/
#include "my_heap.c"
/* 元素入堆 */
void pushMinHeap(MaxHeap *maxHeap, int val) {
// 元素取反
push(maxHeap, -val);
}
/* 元素出堆 */
int popMinHeap(MaxHeap *maxHeap) {
// 元素取反
return -pop(maxHeap);
}
/* 访问堆顶元素 */
int peekMinHeap(MaxHeap *maxHeap) {
// 元素取反
return -peek(maxHeap);
}
/* 取出堆中元素 */
int *getMinHeap(MaxHeap *maxHeap) {
// 将堆中所有元素取反并存入 res 数组
int *res = (int *)malloc(maxHeap->size * sizeof(int));
for (int i = 0; i < maxHeap->size; i++) {
res[i] = -maxHeap->data[i];
}
return res;
}
// 基于堆查找数组中最大的 k 个元素的函数
int *topKHeap(int *nums, int sizeNums, int k) {
// 初始化小顶堆
// 请注意:我们将堆中所有元素取反,从而用大顶堆来模拟小顶堆
int empty[0];
MaxHeap *maxHeap = newMaxHeap(empty, 0);
// 将数组的前 k 个元素入堆
for (int i = 0; i < k; i++) {
pushMinHeap(maxHeap, nums[i]);
}
// 从第 k+1 个元素开始,保持堆的长度为 k
for (int i = k; i < sizeNums; i++) {
// 若当前元素大于堆顶元素,则将堆顶元素出堆、当前元素入堆
if (nums[i] > peekMinHeap(maxHeap)) {
popMinHeap(maxHeap);
pushMinHeap(maxHeap, nums[i]);
}
}
int *res = getMinHeap(maxHeap);
// 释放内存
freeMaxHeap(maxHeap);
return res;
}
/* Driver Code */
int main() {
int nums[] = {1, 7, 6, 3, 2};
int k = 3;
int sizeNums = sizeof(nums) / sizeof(nums[0]);
int *res = topKHeap(nums, sizeNums, k);
printf("最大的 %d 个元素为: ", k);
printArray(res, k);
free(res);
return 0;
}

View file

@ -8,6 +8,7 @@
/* 基于堆查找数组中最大的 k 个元素 */ /* 基于堆查找数组中最大的 k 个元素 */
priority_queue<int, vector<int>, greater<int>> topKHeap(vector<int> &nums, int k) { priority_queue<int, vector<int>, greater<int>> topKHeap(vector<int> &nums, int k) {
// 初始化小顶堆
priority_queue<int, vector<int>, greater<int>> heap; priority_queue<int, vector<int>, greater<int>> heap;
// 将数组的前 k 个元素入堆 // 将数组的前 k 个元素入堆
for (int i = 0; i < k; i++) { for (int i = 0; i < k; i++) {

View file

@ -9,6 +9,7 @@ namespace hello_algo.chapter_heap;
public class top_k { public class top_k {
/* 基于堆查找数组中最大的 k 个元素 */ /* 基于堆查找数组中最大的 k 个元素 */
public static PriorityQueue<int, int> TopKHeap(int[] nums, int k) { public static PriorityQueue<int, int> TopKHeap(int[] nums, int k) {
// 初始化小顶堆
PriorityQueue<int, int> heap = new(); PriorityQueue<int, int> heap = new();
// 将数组的前 k 个元素入堆 // 将数组的前 k 个元素入堆
for (int i = 0; i < k; i++) { for (int i = 0; i < k; i++) {

View file

@ -8,7 +8,7 @@ import '../utils/print_util.dart';
/* 基于堆查找数组中最大的 k 个元素 */ /* 基于堆查找数组中最大的 k 个元素 */
MinHeap topKHeap(List<int> nums, int k) { MinHeap topKHeap(List<int> nums, int k) {
// k // k
MinHeap heap = MinHeap(nums.sublist(0, k)); MinHeap heap = MinHeap(nums.sublist(0, k));
// k+1 k // k+1 k
for (int i = k; i < nums.length; i++) { for (int i = k; i < nums.length; i++) {

View file

@ -32,6 +32,7 @@ func (h *minHeap) Top() any {
/* 基于堆查找数组中最大的 k 个元素 */ /* 基于堆查找数组中最大的 k 个元素 */
func topKHeap(nums []int, k int) *minHeap { func topKHeap(nums []int, k int) *minHeap {
// 初始化小顶堆
h := &minHeap{} h := &minHeap{}
heap.Init(h) heap.Init(h)
// 将数组的前 k 个元素入堆 // 将数组的前 k 个元素入堆

View file

@ -12,6 +12,7 @@ import java.util.*;
public class top_k { public class top_k {
/* 基于堆查找数组中最大的 k 个元素 */ /* 基于堆查找数组中最大的 k 个元素 */
static Queue<Integer> topKHeap(int[] nums, int k) { static Queue<Integer> topKHeap(int[] nums, int k) {
// 初始化小顶堆
Queue<Integer> heap = new PriorityQueue<Integer>(); Queue<Integer> heap = new PriorityQueue<Integer>();
// 将数组的前 k 个元素入堆 // 将数组的前 k 个元素入堆
for (int i = 0; i < k; i++) { for (int i = 0; i < k; i++) {

View file

@ -123,33 +123,35 @@ class MaxHeap {
} }
/* Driver Code */ /* Driver Code */
/* 初始化大顶堆 */ if (require.main === module) {
const maxHeap = new MaxHeap([9, 8, 6, 6, 7, 5, 2, 1, 4, 3, 6, 2]); /* 初始化大顶堆 */
console.log('\n输入列表并建堆后'); const maxHeap = new MaxHeap([9, 8, 6, 6, 7, 5, 2, 1, 4, 3, 6, 2]);
maxHeap.print(); console.log('\n输入列表并建堆后');
maxHeap.print();
/* 获取堆顶元素 */ /* 获取堆顶元素 */
let peek = maxHeap.peek(); let peek = maxHeap.peek();
console.log(`\n堆顶元素为 ${peek}`); console.log(`\n堆顶元素为 ${peek}`);
/* 元素入堆 */ /* 元素入堆 */
let val = 7; let val = 7;
maxHeap.push(val); maxHeap.push(val);
console.log(`\n元素 ${val} 入堆后`); console.log(`\n元素 ${val} 入堆后`);
maxHeap.print(); maxHeap.print();
/* 堆顶元素出堆 */ /* 堆顶元素出堆 */
peek = maxHeap.pop(); peek = maxHeap.pop();
console.log(`\n堆顶元素 ${peek} 出堆后`); console.log(`\n堆顶元素 ${peek} 出堆后`);
maxHeap.print(); maxHeap.print();
/* 获取堆大小 */ /* 获取堆大小 */
let size = maxHeap.size(); let size = maxHeap.size();
console.log(`\n堆元素数量为 ${size}`); console.log(`\n堆元素数量为 ${size}`);
/* 判断堆是否为空 */ /* 判断堆是否为空 */
let isEmpty = maxHeap.isEmpty(); let isEmpty = maxHeap.isEmpty();
console.log(`\n堆是否为空 ${isEmpty}`); console.log(`\n堆是否为空 ${isEmpty}`);
}
module.exports = { module.exports = {
MaxHeap, MaxHeap,

View file

@ -6,25 +6,49 @@
const { MaxHeap } = require('./my_heap'); const { MaxHeap } = require('./my_heap');
/* 元素入堆 */
function pushMinHeap(maxHeap, val) {
// 元素取反
maxHeap.push(-val);
}
/* 元素出堆 */
function popMinHeap(maxHeap) {
// 元素取反
return -maxHeap.pop();
}
/* 访问堆顶元素 */
function peekMinHeap(maxHeap) {
// 元素取反
return -maxHeap.peek();
}
/* 取出堆中元素 */
function getMinHeap(maxHeap) {
// 元素取反
return maxHeap.getMaxHeap().map((num) => -num);
}
/* 基于堆查找数组中最大的 k 个元素 */ /* 基于堆查找数组中最大的 k 个元素 */
function topKHeap(nums, k) { function topKHeap(nums, k) {
// 使用大顶堆 MaxHeap ,对数组 nums 取相反数 // 初始化小顶堆
const invertedNums = nums.map((num) => -num); // 请注意:我们将堆中所有元素取反,从而用大顶堆来模拟小顶堆
const maxHeap = new MaxHeap([]);
// 将数组的前 k 个元素入堆 // 将数组的前 k 个元素入堆
const heap = new MaxHeap(invertedNums.slice(0, k)); for (let i = 0; i < k; i++) {
pushMinHeap(maxHeap, nums[i]);
}
// 从第 k+1 个元素开始,保持堆的长度为 k // 从第 k+1 个元素开始,保持堆的长度为 k
for (let i = k; i < invertedNums.length; i++) { for (let i = k; i < nums.length; i++) {
// 若当前元素小于堆顶元素,则将堆顶元素出堆、当前元素入堆 // 若当前元素于堆顶元素,则将堆顶元素出堆、当前元素入堆
if (invertedNums[i] < heap.peek()) { if (nums[i] > peekMinHeap(maxHeap)) {
heap.pop(); popMinHeap(maxHeap);
heap.push(invertedNums[i]); pushMinHeap(maxHeap, nums[i]);
} }
} }
// 取出堆中元素 // 返回堆中元素
const maxHeap = heap.getMaxHeap(); return getMinHeap(maxHeap);
// 对堆中元素取相反数
const invertedMaxHeap = maxHeap.map((num) => -num);
return invertedMaxHeap;
} }
/* Driver Code */ /* Driver Code */

View file

@ -15,6 +15,7 @@ import heapq
def top_k_heap(nums: list[int], k: int) -> list[int]: def top_k_heap(nums: list[int], k: int) -> list[int]:
"""基于堆查找数组中最大的 k 个元素""" """基于堆查找数组中最大的 k 个元素"""
# 初始化小顶堆
heap = [] heap = []
# 将数组的前 k 个元素入堆 # 将数组的前 k 个元素入堆
for i in range(k): for i in range(k):

View file

@ -11,7 +11,7 @@ use std::collections::BinaryHeap;
/* 基于堆查找数组中最大的 k 个元素 */ /* 基于堆查找数组中最大的 k 个元素 */
fn top_k_heap(nums: Vec<i32>, k: usize) -> BinaryHeap<Reverse<i32>> { fn top_k_heap(nums: Vec<i32>, k: usize) -> BinaryHeap<Reverse<i32>> {
// Rust 的 BinaryHeap 是大顶堆,使用 Reverse 将元素大小反转 // BinaryHeap 是大顶堆,使用 Reverse 将元素取反,从而实现小顶堆
let mut heap = BinaryHeap::<Reverse<i32>>::new(); let mut heap = BinaryHeap::<Reverse<i32>>::new();
// 将数组的前 k 个元素入堆 // 将数组的前 k 个元素入堆
for &num in nums.iter().take(k) { for &num in nums.iter().take(k) {