1. 题目

2. 解答

2.1. 方法一——大顶堆

参考 堆和堆排序 以及 堆的应用,我们将数组的前 K 个位置当作一个大顶堆。

首先建堆,也即对堆中 [0, (K-2)/2] 的节点从上往下进行堆化。第 K/2 个节点若有子节点,其左子节点位置应该为 2 * K/2 + 1 = K+1,而我们堆中的最大位置为 K-1,显然第 K/2 个节点是第一个叶子节点,不用堆化。

建完堆之后,我们顺序访问原数组 [k, n-1] 位置的元素,如果当前元素小于堆顶元素也就是位置为 0 的元素,那么删除堆顶元素并将当前元素插入堆中。

最后,堆中的 K 个元素即为所求。

class Solution {
public:
vector<int> GetLeastNumbers_Solution(vector<int> input, int k) { if (k > input.size())
{
vector<int> result;
return result;
}
Build_Heap_K(input, k);
vector<int> result(input.begin(), input.begin()+k);
return result;
} void Build_Heap_K(vector<int> &input, int k)
{
// input 的 [0, k-1] 作为一个大小为 K 的大顶堆
// 然后从上往下进行堆化
// 也就是对堆中 [0, (k-2)/2] 的节点进行堆化
for (int i = (k-2)/2; i >= 0; i--)
Heapify(input, k, i); // 遍历 input 的 [k, n-1] 的元素
// 如果某元素小于堆顶值,将其插入堆中
// 也即将其替换为堆顶元素,堆化之
for (int i = k; i < input.size(); i++)
{
if (input[i] < input[0])
{
input[0] = input[i];
Heapify(input, k, 0);
}
}
} void Heapify(vector<int> &input, int k, int i)
{
while(1)
{
int max_pos = i;
if (2*i+1 < k && input[2*i+1] > input[max_pos])
max_pos = 2 * i + 1;
if (2*i+2 < k && input[2*i+2] > input[max_pos])
max_pos = 2 * i + 2;
if (max_pos == i)
break;
else
{
int temp = input[max_pos];
input[max_pos] = input[i];
input[i] = temp;
}
i = max_pos;
}
}
};
2.2. 方法二——快排分治

可参考 LeetCode 215——数组中的第 K 个最大元素

快排的时候需要分区,分区点左边的元素都小于主元,分区点右边的元素都大于主元。如果分区后主元的位置恰好为 K,那左边正好是最小的 K 个数;如果大于 K,我们需要递归在左边找到第 K 个位置;如果小于 K,我们则需要递归在右边找到第 K 个位置。

class Solution {
public:
vector<int> GetLeastNumbers_Solution(vector<int> input, int k) {
int n = input.size();
if (k > n)
{
vector<int> result;
return result;
}
Quick_Sort(input, 0, n-1, k);
vector<int> result(input.begin(), input.begin()+k);
return result;
} void Quick_Sort(vector<int> &input, int left, int right, int k)
{
if (left < right)
{
int pivot = input[right];
int i = left;
int j = left; for (; j < right; j++)
{
if (input[j] < pivot)
{
int temp = input[i];
input[i] = input[j];
input[j] = temp;
i++;
}
}
input[j] = input[i];
input[i] = pivot; if (i == k) return;
else if (i > k) Quick_Sort(input, left, i-1, k);
else Quick_Sort(input, i+1, right, k);
}
}
};

获取更多精彩,请关注「seniusen」!

剑指 Offer——最小的 K 个数的更多相关文章

  1. 剑指offer 最小的k个数 、 leetcode 215. Kth Largest Element in an Array 、295. Find Median from Data Stream(剑指 数据流中位数)

    注意multiset的一个bug: multiset带一个参数的erase函数原型有两种.一是传递一个元素值,如上面例子代码中,这时候删除的是集合中所有值等于输入值的元素,并且返回删除的元素个数:另外 ...

  2. 剑指Offer——最小的K个数

    题目描述: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4. 分析: 建一个K大小的大根堆,存储最小的k个数字. 先将K个数进堆 ...

  3. python剑指offer最小的K个数

    题目描述: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 思路: 使用快排中的partition思想. ①我们设定part ...

  4. 用js刷剑指offer(最小的K个数)

    题目描述 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 牛客网链接 js代码 function GetLeastNumbe ...

  5. 剑指offer--10.最小的K个数

    边界判断,坑了一下 ----------------------------------------------- 时间限制:1秒 空间限制:32768K 热度指数:375643 本题知识点: 数组 ...

  6. 剑指Offer-29.最小的K个数(C++/Java)

    题目: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 分析: 最先想到的是将数组升序排列,返回前k个元素.不过排序的话效率 ...

  7. 剑指:最小的k个数

    题目描述 输入 n 个整数,找出其中最小的 K 个数.例如输入 4,5,1,6,2,7,3,8 这 8 个数字,则最小的 4 个数字是 1,2,3,4. 解法 解法一 利用快排中的 partition ...

  8. 2-剑指offer: 最小的K个数

    题目描述 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 代码: // 这种topN问题比较常见的是使用堆来解决,最小的k个 ...

  9. 剑指Offer28 最小的K个数(Partition函数应用+大顶堆)

    包含了Partition函数的多种用法 以及大顶堆操作 /*********************************************************************** ...

随机推荐

  1. You-Get,多网站视频下载工具,非常方便

    You-Get是一个非常优秀的网站视频下载工具.使用You-Get可以很轻松的下载到网络上的视频.图片及音乐. 按Win+R键打开运行,输入cmd,再输入命令 pip install you-get, ...

  2. Codeforces Round #503 (by SIS, Div. 2) E. Sergey's problem

    E. Sergey's problem [题目描述] 给出一个n个点m条边的有向图,需要找到一个集合使得1.集合中的各点之间无无边相连2.集合外的点到集合内的点的最小距离小于等于2. [算法] 官方题 ...

  3. BZOJ 5317: [Jsoi2018]部落战争

    传送门 写出式子,若存在 $a \in A$,$b \in B$,使得 $b+v=a$,那么此方案会产生冲突 即存在 $a \in A$,$b \in B$,使得 $v=a+(-b)$,设 $C=A+ ...

  4. 在Visual studio 2017中使用EF6连接MySQL

    在Visual studio 2017中使用EF6连接Mysql ADO.NET Entity Framework 是微软以 ADO.NET 为基础所发展出来的对象关系对应 (O/R Mapping) ...

  5. Python爬虫之urllib.parse详解

    Python爬虫之urllib.parse 转载地址 Python 中的 urllib.parse 模块提供了很多解析和组建 URL 的函数. 解析url 解析url( urlparse() ) ur ...

  6. logname - 显示用户登录名

    总览 (SYNOPSIS) logname [OPTION]... 描述 (DESCRIPTION) 显示 当前用户 的 名字. --help 显示 帮助信息, 然后 结束. --version 显示 ...

  7. 03python面向对象编程1

    1.创建和使用类 1.1 创建 Dog 类.根据 Dog 类创建的每个实例都将存储名字和年龄.我们赋予了每条小狗蹲下( sit() )和打滚( roll_over() )的能力: In [2]: cl ...

  8. Crazy Search POJ - 1200 (字符串哈希hash)

    Many people like to solve hard puzzles some of which may lead them to madness. One such puzzle could ...

  9. CTF各种资源:题目、工具、资料

    目录 题目汇总 Reverse 签到题 Web Web中等难度 Crypto 基础网站 各类工具 综合 Web Payloads 逆向 Pwn 取证 题目汇总 这里收集了我做过的CTF题目 Rever ...

  10. 我用过的gitlab api

    1.新增tag https://docs.gitlab.com/ee/api/tags.html#list-project-repository-tags 2.获取指定项目合分支的最新一次merge ...