剑指 Offer 40. 最小的k个数 - 力扣(LeetCode)

输入整数数组 arr ,找出其中最小的 k 个数。例如,输入4、5、1、6、2、7、3、8这8个数字,则最小的4个数字是1、2、3、4。

思路:


  1. 快速划分

    image-20221112191102137

  2. 大根堆

    我们用一个大根堆实时维护数组的前 k 小值。首先将前 k 个数插入大根堆中,随后从第 k+1 个数开始遍历,如果当前遍历到的数比大根堆的堆顶的数要小,就把堆顶的数弹出,再插入当前遍历到的数。最后将大根堆里的数存入数组返回即可。

复杂度:

O(k)

题解:

  1. 大根堆
class Solution {
public:
vector<int> getLeastNumbers(vector<int>& arr, int k) {
vector<int> vec(k, 0);
if (k == 0) { // 排除 0 的情况
return vec;
}
priority_queue<int> Q;
for (int i = 0; i < k; ++i) {
Q.push(arr[i]);
}
for (int i = k; i < (int)arr.size(); ++i) {//每次pop出最大的一s个数
if (Q.top() > arr[i]) {
Q.pop();
Q.push(arr[i]);
}
}
for (int i = 0; i < k; ++i) {
vec[i] = Q.top();
Q.pop();
}
return vec;
}
};

  1. 快速划分
class Solution {
int partition(vector<int>& nums, int l, int r) {
int pivot = nums[r];
int i = l - 1;
for (int j = l; j <= r - 1; ++j) {
if (nums[j] <= pivot) {//将《=基准值的数放在基准值左边
i = i + 1;
swap(nums[i], nums[j]);
}
}
swap(nums[i + 1], nums[r]);//将基准值换到中间
return i + 1;
}

// 基于随机的划分
int randomized_partition(vector<int>& nums, int l, int r) {

int i = rand() % (r - l + 1) + l;//随机选择一个作为快速划分的基准值
swap(nums[r], nums[i]);//方便下个函数取值
return partition(nums, l, r);
}

//划分数组 arr 的 [l,r] 部分,使前 k 小的数在数组的左侧
void randomized_selected(vector<int>& arr, int l, int r, int k) {
if (l >= r) {
return;
}

//分界值在数组中的位置
int pos = randomized_partition(arr, l, r);
//已经前num小的数字放置在【l,pos】区间
int num = pos - l + 1;

if (k == num) {//符合要求,结束
return;
} else if (k < num) {//区间划得太大了,缩小区间
randomized_selected(arr, l, pos - 1, k);
} else {//太小了,扩大区间
randomized_selected(arr, pos + 1, r, k - num);
}
}

public:
vector<int> getLeastNumbers(vector<int>& arr, int k) {
srand((unsigned)time(NULL));
//入口
randomized_selected(arr, 0, (int)arr.size() - 1, k);
vector<int> vec;
for (int i = 0; i < k; ++i) {
vec.push_back(arr[i]);
}
return vec;
}
};

作者:LeetCode-Solution
链接:https://leetcode.cn/problems/zui-xiao-de-kge-shu-lcof/solution/zui-xiao-de-kge-shu-by-leetcode-solution/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。